VirtualBox

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

Last change on this file since 59466 was 58170, checked in by vboxsync, 9 years ago

doxygen: fixes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 55.5 KB
Line 
1/* $Id: USBProxyDevice-usbip.cpp 58170 2015-10-12 09:27:14Z 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(0x0111)
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(3)
59/** URB unlink (cancel) command identifier. */
60#define USBIP_CMD_UNLINK UINT32_C(2)
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(1)
75/** URB direction - output. */
76#define USBIP_DIR_OUT UINT32_C(0)
77
78/** @name USB/IP error codes.
79 * @{ */
80/** Success indicator. */
81#define USBIP_STATUS_SUCCESS INT32_C(0)
82/** Pipe stalled. */
83#define USBIP_STATUS_PIPE_STALLED INT32_C(-32)
84/** Short read. */
85#define USBIP_STATUS_SHORT_READ INT32_C(-121)
86/** @} */
87
88/**
89 * Exported device entry in the OP_RET_DEVLIST reply.
90 */
91#pragma pack(1)
92typedef struct UsbIpExportedDevice
93{
94 /** Path of the device, zero terminated string. */
95 char szPath[256];
96 /** Bus ID of the exported device, zero terminated string. */
97 char szBusId[32];
98 /** Bus number. */
99 uint32_t u32BusNum;
100 /** Device number. */
101 uint32_t u32DevNum;
102 /** Speed indicator of the device. */
103 uint32_t u32Speed;
104 /** Vendor ID of the device. */
105 uint16_t u16VendorId;
106 /** Product ID of the device. */
107 uint16_t u16ProductId;
108 /** Device release number. */
109 uint16_t u16BcdDevice;
110 /** Device class. */
111 uint8_t bDeviceClass;
112 /** Device Subclass. */
113 uint8_t bDeviceSubClass;
114 /** Device protocol. */
115 uint8_t bDeviceProtocol;
116 /** Configuration value. */
117 uint8_t bConfigurationValue;
118 /** Current configuration value of the device. */
119 uint8_t bNumConfigurations;
120 /** Number of interfaces for the device. */
121 uint8_t bNumInterfaces;
122} UsbIpExportedDevice;
123/** Pointer to a exported device entry. */
124typedef UsbIpExportedDevice *PUsbIpExportedDevice;
125#pragma pack()
126AssertCompileSize(UsbIpExportedDevice, 312);
127
128/**
129 * Interface descriptor entry for an exported device.
130 */
131#pragma pack(1)
132typedef struct UsbIpDeviceInterface
133{
134 /** Intefrace class. */
135 uint8_t bInterfaceClass;
136 /** Interface sub class. */
137 uint8_t bInterfaceSubClass;
138 /** Interface protocol identifier. */
139 uint8_t bInterfaceProtocol;
140 /** Padding byte for alignment. */
141 uint8_t bPadding;
142} UsbIpDeviceInterface;
143/** Pointer to an interface descriptor entry. */
144typedef UsbIpDeviceInterface *PUsbIpDeviceInterface;
145#pragma pack()
146
147/**
148 * USB/IP Import request.
149 */
150#pragma pack(1)
151typedef struct UsbIpReqImport
152{
153 /** Protocol version number. */
154 uint16_t u16Version;
155 /** Command code. */
156 uint16_t u16Cmd;
157 /** Status field, unused. */
158 int32_t u32Status;
159 /** Bus Id of the device as zero terminated string. */
160 char aszBusId[32];
161} UsbIpReqImport;
162/** Pointer to a import request. */
163typedef UsbIpReqImport *PUsbIpReqImport;
164#pragma pack()
165
166/**
167 * USB/IP Import reply.
168 *
169 * This is only the header, for successful
170 * imports the device details are sent to as
171 * defined in UsbIpExportedDevice.
172 */
173#pragma pack(1)
174typedef struct UsbIpRetImport
175{
176 /** Protocol version number. */
177 uint16_t u16Version;
178 /** Command code. */
179 uint16_t u16Cmd;
180 /** Status field, unused. */
181 int32_t u32Status;
182} UsbIpRetImport;
183/** Pointer to a import reply. */
184typedef UsbIpRetImport *PUsbIpRetImport;
185#pragma pack()
186
187/**
188 * Command/Reply header common to the submit and unlink commands
189 * replies.
190 */
191#pragma pack(1)
192typedef struct UsbIpReqRetHdr
193{
194 /** Request/Return code. */
195 uint32_t u32ReqRet;
196 /** Sequence number to identify the URB. */
197 uint32_t u32SeqNum;
198 /** Device id. */
199 uint32_t u32DevId;
200 /** Direction of the endpoint (host->device, device->host). */
201 uint32_t u32Direction;
202 /** Endpoint number. */
203 uint32_t u32Endpoint;
204} UsbIpReqRetHdr;
205/** Pointer to a request/reply header. */
206typedef UsbIpReqRetHdr *PUsbIpReqRetHdr;
207#pragma pack()
208
209/**
210 * USB/IP Submit request.
211 */
212#pragma pack(1)
213typedef struct UsbIpReqSubmit
214{
215 /** The request header. */
216 UsbIpReqRetHdr Hdr;
217 /** Transfer flags for the URB. */
218 uint32_t u32XferFlags;
219 /** Transfer buffer length. */
220 uint32_t u32TransferBufferLength;
221 /** Frame to transmit an ISO frame. */
222 uint32_t u32StartFrame;
223 /** Number of isochronous packets. */
224 uint32_t u32NumIsocPkts;
225 /** Maximum time for the request on the server side host controller. */
226 uint32_t u32Interval;
227 /** Setup data for a control URB. */
228 VUSBSETUP Setup;
229} UsbIpReqSubmit;
230/** Pointer to a submit request. */
231typedef UsbIpReqSubmit *PUsbIpReqSubmit;
232#pragma pack()
233
234/**
235 * USB/IP Submit reply.
236 */
237#pragma pack(1)
238typedef struct UsbIpRetSubmit
239{
240 /** The reply header. */
241 UsbIpReqRetHdr Hdr;
242 /** Status code. */
243 int32_t u32Status;
244 /** Actual length of the reply buffer. */
245 uint32_t u32ActualLength;
246 /** The actual selected frame for a isochronous transmit. */
247 uint32_t u32StartFrame;
248 /** Number of isochronous packets. */
249 uint32_t u32NumIsocPkts;
250 /** Number of failed isochronous packets. */
251 uint32_t u32ErrorCount;
252 /** Setup data for a control URB. */
253 VUSBSETUP Setup;
254} UsbIpRetSubmit;
255/** Pointer to a submit reply. */
256typedef UsbIpRetSubmit *PUsbIpRetSubmit;
257#pragma pack()
258
259/**
260 * Unlink URB request.
261 */
262#pragma pack(1)
263typedef struct UsbIpReqUnlink
264{
265 /** The request header. */
266 UsbIpReqRetHdr Hdr;
267 /** The sequence number to unlink. */
268 uint32_t u32SeqNum;
269} UsbIpReqUnlink;
270/** Pointer to a URB unlink request. */
271typedef UsbIpReqUnlink *PUsbIpReqUnlink;
272#pragma pack()
273
274/**
275 * Unlink URB reply.
276 */
277#pragma pack(1)
278typedef struct UsbIpRetUnlink
279{
280 /** The reply header. */
281 UsbIpReqRetHdr Hdr;
282 /** Status of the request. */
283 int32_t u32Status;
284} UsbIpRetUnlink;
285/** Pointer to a URB unlink request. */
286typedef UsbIpRetUnlink *PUsbIpRetUnlink;
287#pragma pack()
288
289/**
290 * Union of possible replies from the server during normal operation.
291 */
292#pragma pack(1)
293typedef union UsbIpRet
294{
295 /** The header. */
296 UsbIpReqRetHdr Hdr;
297 /** Submit reply. */
298 UsbIpRetSubmit RetSubmit;
299 /** Unlink reply. */
300 UsbIpRetUnlink RetUnlink;
301 /** Byte view. */
302 uint8_t abReply[1];
303} UsbIpRet;
304/** Pointer to a reply union. */
305typedef UsbIpRet *PUsbIpRet;
306#pragma pack()
307
308/**
309 * Isochronous packet descriptor.
310*/
311#pragma pack(1)
312typedef struct UsbIpIsocPktDesc
313{
314 /** Offset */
315 uint32_t u32Offset;
316 /** Length of the packet including padding. */
317 uint32_t u32Length;
318 /** Size of the transmitted data. */
319 uint32_t u32ActualLength;
320 /** Completion status for this packet. */
321 int32_t i32Status;
322} UsbIpIsocPktDesc;
323/** Pointer to a isochronous packet descriptor. */
324typedef UsbIpIsocPktDesc *PUsbIpIsocPktDesc;
325#pragma pack()
326
327/**
328 * USB/IP backend specific data for one URB.
329 * Required for tracking in flight and landed URBs.
330 */
331typedef struct USBPROXYURBUSBIP
332{
333 /** List node for the in flight or landed URB list. */
334 RTLISTNODE NodeList;
335 /** Sequence number the assigned URB is identified by. */
336 uint32_t u32SeqNumUrb;
337 /** Pointer to the VUSB URB. */
338 PVUSBURB pVUsbUrb;
339} USBPROXYURBUSBIP;
340/** Pointer to a USB/IP URB. */
341typedef USBPROXYURBUSBIP *PUSBPROXYURBUSBIP;
342
343/**
344 * USB/IP data receive states.
345 */
346typedef enum USBPROXYUSBIPRECVSTATE
347{
348 /** Invalid receive state. */
349 USBPROXYUSBIPRECVSTATE_INVALID = 0,
350 /** Currently receiving the common header structure. */
351 USBPROXYUSBIPRECVSTATE_HDR_COMMON,
352 /** Currently receieving the rest of the header structure. */
353 USBPROXYUSBIPRECVSTATE_HDR_RESIDUAL,
354 /** Currently receiving data into the URB buffer. */
355 USBPROXYUSBIPRECVSTATE_URB_BUFFER,
356 /** Currently receiving the isochronous packet descriptors. */
357 USBPROXYUSBIPRECVSTATE_ISOC_PKT_DESCS,
358 /** Usual 32bit hack. */
359 USBPROXYUSBIPRECVSTATE_32BIT_HACK = 0x7fffffff
360} USBPROXYUSBIPRECVSTATE;
361/** Pointer to an receive state. */
362typedef USBPROXYUSBIPRECVSTATE *PUSBPROXYUSBIPRECVSTATE;
363
364/**
365 * Backend data for the USB/IP USB Proxy device backend.
366 */
367typedef struct USBPROXYDEVUSBIP
368{
369 /** IPRT socket handle. */
370 RTSOCKET hSocket;
371 /** Pollset with the wakeup pipe and socket. */
372 RTPOLLSET hPollSet;
373 /** Pipe endpoint - read (in the pollset). */
374 RTPIPE hPipeR;
375 /** Pipe endpoint - write. */
376 RTPIPE hPipeW;
377 /** Next sequence number to use for identifying submitted URBs. */
378 volatile uint32_t u32SeqNumNext;
379 /** Fast mutex protecting the lists below against concurrent access. */
380 RTSEMFASTMUTEX hMtxLists;
381 /** List of in flight URBs. */
382 RTLISTANCHOR ListUrbsInFlight;
383 /** List of landed URBs. */
384 RTLISTANCHOR ListUrbsLanded;
385 /** List of URBs to submit. */
386 RTLISTANCHOR ListUrbsToQueue;
387 /** Port of the USB/IP host to connect to. */
388 uint32_t uPort;
389 /** USB/IP host address. */
390 char *pszHost;
391 /** USB Bus ID of the device to capture. */
392 char *pszBusId;
393 /** The device ID to use to identify the device. */
394 uint32_t u32DevId;
395 /** Temporary buffer for the next reply header */
396 UsbIpRet BufRet;
397 /** Temporary buffer to hold all isochronous packet descriptors. */
398 UsbIpIsocPktDesc aIsocPktDesc[8];
399 /** Pointer to the current buffer to write received data to. */
400 uint8_t *pbRecv;
401 /** Number of bytes received so far. */
402 size_t cbRecv;
403 /** Number of bytes left to receive. until we advance the state machine and process the data */
404 size_t cbLeft;
405 /** The current receiving state. */
406 USBPROXYUSBIPRECVSTATE enmRecvState;
407 /** The URB we currently receive a response for. */
408 PUSBPROXYURBUSBIP pUrbUsbIp;
409} USBPROXYDEVUSBIP, *PUSBPROXYDEVUSBIP;
410
411/** Pollset id of the socket. */
412#define USBIP_POLL_ID_SOCKET 0
413/** Pollset id of the pipe. */
414#define USBIP_POLL_ID_PIPE 1
415
416/** USB/IP address prefix for identifcation. */
417#define USBIP_URI_PREFIX "usbip://"
418/** USB/IP address prefix length. */
419#define USBIP_URI_PREFIX_LEN (sizeof(USBIP_URI_PREFIX) - 1)
420
421/** Waking reason for the USB I/P reaper: New URBs to queue. */
422#define USBIP_REAPER_WAKEUP_REASON_QUEUE 'Q'
423/** Waking reason for the USB I/P reaper: External wakeup. */
424#define USBIP_REAPER_WAKEUP_REASON_EXTERNAL 'E'
425
426/**
427 * Converts a request/reply header from network to host endianness.
428 *
429 * @returns nothing.
430 * @param pHdr The header to convert.
431 */
432DECLINLINE(void) usbProxyUsbIpReqRetHdrN2H(PUsbIpReqRetHdr pHdr)
433{
434 pHdr->u32ReqRet = RT_H2N_U32(pHdr->u32ReqRet);
435 pHdr->u32SeqNum = RT_H2N_U32(pHdr->u32SeqNum);
436 pHdr->u32DevId = RT_H2N_U32(pHdr->u32DevId);
437 pHdr->u32Direction = RT_H2N_U32(pHdr->u32Direction);
438 pHdr->u32Endpoint = RT_H2N_U32(pHdr->u32Endpoint);
439}
440
441/**
442 * Converts a request/reply header from host to network endianness.
443 *
444 * @returns nothing.
445 * @param pHdr The header to convert.
446 */
447DECLINLINE(void) usbProxyUsbIpReqRetHdrH2N(PUsbIpReqRetHdr pHdr)
448{
449 pHdr->u32ReqRet = RT_N2H_U32(pHdr->u32ReqRet);
450 pHdr->u32SeqNum = RT_N2H_U32(pHdr->u32SeqNum);
451 pHdr->u32DevId = RT_N2H_U32(pHdr->u32DevId);
452 pHdr->u32Direction = RT_N2H_U32(pHdr->u32Direction);
453 pHdr->u32Endpoint = RT_N2H_U32(pHdr->u32Endpoint);
454}
455
456/**
457 * Converts a submit request from host to network endianness.
458 *
459 * @returns nothing.
460 * @param pReqSubmit The submit request to convert.
461 */
462DECLINLINE(void) usbProxyUsbIpReqSubmitH2N(PUsbIpReqSubmit pReqSubmit)
463{
464 usbProxyUsbIpReqRetHdrH2N(&pReqSubmit->Hdr);
465 pReqSubmit->u32XferFlags = RT_H2N_U32(pReqSubmit->u32XferFlags);
466 pReqSubmit->u32TransferBufferLength = RT_H2N_U32(pReqSubmit->u32TransferBufferLength);
467 pReqSubmit->u32StartFrame = RT_H2N_U32(pReqSubmit->u32StartFrame);
468 pReqSubmit->u32NumIsocPkts = RT_H2N_U32(pReqSubmit->u32NumIsocPkts);
469 pReqSubmit->u32Interval = RT_H2N_U32(pReqSubmit->u32Interval);
470}
471
472/**
473 * Converts a submit reply from network to host endianness.
474 *
475 * @returns nothing.
476 * @param pReqSubmit The submit reply to convert.
477 */
478DECLINLINE(void) usbProxyUsbIpRetSubmitN2H(PUsbIpRetSubmit pRetSubmit)
479{
480 usbProxyUsbIpReqRetHdrN2H(&pRetSubmit->Hdr);
481 pRetSubmit->u32Status = RT_N2H_U32(pRetSubmit->u32Status);
482 pRetSubmit->u32ActualLength = RT_N2H_U32(pRetSubmit->u32ActualLength);
483 pRetSubmit->u32StartFrame = RT_N2H_U32(pRetSubmit->u32StartFrame);
484 pRetSubmit->u32NumIsocPkts = RT_N2H_U32(pRetSubmit->u32NumIsocPkts);
485 pRetSubmit->u32ErrorCount = RT_N2H_U32(pRetSubmit->u32ErrorCount);
486}
487
488/**
489 * Converts a isochronous packet descriptor from host to network endianness.
490 *
491 * @returns nothing.
492 * @param pIsocPktDesc The packet descriptor to convert.
493 */
494DECLINLINE(void) usbProxyUsbIpIsocPktDescH2N(PUsbIpIsocPktDesc pIsocPktDesc)
495{
496 pIsocPktDesc->u32Offset = RT_H2N_U32(pIsocPktDesc->u32Offset);
497 pIsocPktDesc->u32Length = RT_H2N_U32(pIsocPktDesc->u32Length);
498 pIsocPktDesc->u32ActualLength = RT_H2N_U32(pIsocPktDesc->u32ActualLength);
499 pIsocPktDesc->i32Status = RT_H2N_U32(pIsocPktDesc->i32Status);
500}
501
502/**
503 * Converts a isochronous packet descriptor from network to host endianness.
504 *
505 * @returns nothing.
506 * @param pIsocPktDesc The packet descriptor to convert.
507 */
508DECLINLINE(void) usbProxyUsbIpIsocPktDescN2H(PUsbIpIsocPktDesc pIsocPktDesc)
509{
510 pIsocPktDesc->u32Offset = RT_N2H_U32(pIsocPktDesc->u32Offset);
511 pIsocPktDesc->u32Length = RT_N2H_U32(pIsocPktDesc->u32Length);
512 pIsocPktDesc->u32ActualLength = RT_N2H_U32(pIsocPktDesc->u32ActualLength);
513 pIsocPktDesc->i32Status = RT_N2H_U32(pIsocPktDesc->i32Status);
514}
515
516/**
517 * Converts a unlink request from host to network endianness.
518 *
519 * @returns nothing.
520 * @param pReqUnlink The unlink request to convert.
521 */
522DECLINLINE(void) usbProxyUsbIpReqUnlinkH2N(PUsbIpReqUnlink pReqUnlink)
523{
524 usbProxyUsbIpReqRetHdrH2N(&pReqUnlink->Hdr);
525 pReqUnlink->u32SeqNum = RT_H2N_U32(pReqUnlink->u32SeqNum);
526}
527
528/**
529 * Converts a unlink reply from network to host endianness.
530 *
531 * @returns nothing.
532 * @param pRetUnlink The unlink reply to convert.
533 */
534DECLINLINE(void) usbProxyUsbIpRetUnlinkN2H(PUsbIpRetUnlink pRetUnlink)
535{
536 usbProxyUsbIpReqRetHdrN2H(&pRetUnlink->Hdr);
537 pRetUnlink->u32Status = RT_N2H_U32(pRetUnlink->u32Status);
538}
539
540/**
541 * Convert the given exported device structure from host to network byte order.
542 *
543 * @returns nothing.
544 * @param pDevice The device structure to convert.
545 */
546DECLINLINE(void) usbProxyUsbIpExportedDeviceN2H(PUsbIpExportedDevice pDevice)
547{
548 pDevice->u32BusNum = RT_N2H_U32(pDevice->u32BusNum);
549 pDevice->u32DevNum = RT_N2H_U32(pDevice->u32DevNum);
550 pDevice->u32Speed = RT_N2H_U16(pDevice->u32Speed);
551 pDevice->u16VendorId = RT_N2H_U16(pDevice->u16VendorId);
552 pDevice->u16ProductId = RT_N2H_U16(pDevice->u16ProductId);
553 pDevice->u16BcdDevice = RT_N2H_U16(pDevice->u16BcdDevice);
554}
555
556/**
557 * Converts a USB/IP status code to a VBox status code.
558 *
559 * @returns VUSB status code.
560 * @param i32Status The USB/IP status code from the reply.
561 */
562DECLINLINE(int) usbProxyUsbIpStatusConvertFromStatus(int32_t i32Status)
563{
564 if (RT_LIKELY( i32Status == USBIP_STATUS_SUCCESS
565 || i32Status == USBIP_STATUS_SHORT_READ))
566 return VINF_SUCCESS;
567
568 switch (i32Status)
569 {
570 case USBIP_STATUS_PIPE_STALLED:
571 return VINF_SUCCESS;
572 default:
573 return VERR_INVALID_STATE;
574 }
575
576 return VERR_INVALID_STATE;
577}
578
579/**
580 * Converts a USB/IP status code to a VUSB status code.
581 *
582 * @returns VUSB status code.
583 * @param i32Status The USB/IP status code from the reply.
584 */
585DECLINLINE(VUSBSTATUS) usbProxyUsbIpVUsbStatusConvertFromStatus(int32_t i32Status)
586{
587 if (RT_LIKELY( i32Status == USBIP_STATUS_SUCCESS
588 || i32Status == USBIP_STATUS_SHORT_READ))
589 return VUSBSTATUS_OK;
590
591 switch (i32Status)
592 {
593 case USBIP_STATUS_PIPE_STALLED:
594 return VUSBSTATUS_STALL;
595 default:
596 return VUSBSTATUS_DNR;
597 }
598
599 return VUSBSTATUS_DNR;
600}
601
602/**
603 * Gets the next free sequence number.
604 *
605 * @returns Next free sequence number.
606 * @param pProxyDevUsbIp The USB/IP proxy device data.
607 */
608DECLINLINE(uint32_t) usbProxyUsbIpSeqNumGet(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
609{
610 return ASMAtomicIncU32(&pProxyDevUsbIp->u32SeqNumNext);
611}
612
613/**
614 * Links a given URB into the given list.
615 *
616 * @returns nothing.
617 * @param pProxyDevUsbIp The USB/IP proxy device data.
618 * @param pList The list to link the URB into.
619 * @param pUrbUsbIp The URB to link.
620 */
621DECLINLINE(void) usbProxyUsbIpLinkUrb(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PRTLISTANCHOR pList, PUSBPROXYURBUSBIP pUrbUsbIp)
622{
623 int rc = RTSemFastMutexRequest(pProxyDevUsbIp->hMtxLists);
624 AssertRC(rc);
625 RTListAppend(pList, &pUrbUsbIp->NodeList);
626 RTSemFastMutexRelease(pProxyDevUsbIp->hMtxLists);
627}
628
629/**
630 * Unlinks a given URB from the current assigned list.
631 *
632 * @returns nothing.
633 * @param pProxyDevUsbIp The USB/IP proxy device data.
634 * @param pUrbUsbIp The URB to unlink.
635 */
636DECLINLINE(void) usbProxyUsbIpUnlinkUrb(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PUSBPROXYURBUSBIP pUrbUsbIp)
637{
638 int rc = RTSemFastMutexRequest(pProxyDevUsbIp->hMtxLists);
639 AssertRC(rc);
640 RTListNodeRemove(&pUrbUsbIp->NodeList);
641 RTSemFastMutexRelease(pProxyDevUsbIp->hMtxLists);
642}
643
644/**
645 * Allocates a USB/IP proxy specific URB state.
646 *
647 * @returns Pointer to the USB/IP specific URB data or NULL on failure.
648 * @param pProxyDevUsbIp The USB/IP proxy device data.
649 */
650static PUSBPROXYURBUSBIP usbProxyUsbIpUrbAlloc(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
651{
652 NOREF(pProxyDevUsbIp);
653 return (PUSBPROXYURBUSBIP)RTMemAllocZ(sizeof(USBPROXYURBUSBIP));
654}
655
656/**
657 * Frees the given USB/IP URB state.
658 *
659 * @returns nothing.
660 * @param pProxyDevUsbIp The USB/IP proxy device data.
661 * @param pUrbUsbIp The USB/IP speciic URB data.
662 */
663static void usbProxyUsbIpUrbFree(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PUSBPROXYURBUSBIP pUrbUsbIp)
664{
665 NOREF(pProxyDevUsbIp);
666 RTMemFree(pUrbUsbIp);
667}
668
669/**
670 * Parse the string representation of the host address.
671 *
672 * @returns VBox status code.
673 * @param pProxyDevUsbIp The USB/IP proxy device data to parse the address for.
674 * @param pszAddress The address string to parse.
675 */
676static int usbProxyUsbIpParseAddress(PUSBPROXYDEVUSBIP pProxyDevUsbIp, const char *pszAddress)
677{
678 int rc = VINF_SUCCESS;
679
680 if (!RTStrNCmp(pszAddress, USBIP_URI_PREFIX, USBIP_URI_PREFIX_LEN))
681 {
682 pszAddress += USBIP_URI_PREFIX_LEN;
683
684 const char *pszPortStart = RTStrStr(pszAddress, ":");
685 if (pszPortStart)
686 {
687 pszPortStart++;
688
689 const char *pszBusIdStart = RTStrStr(pszPortStart, ":");
690 if (pszBusIdStart)
691 {
692 size_t cbHost = pszPortStart - pszAddress - 1;
693 size_t cbBusId = strlen(pszBusIdStart);
694
695 pszBusIdStart++;
696
697 rc = RTStrToUInt32Ex(pszPortStart, NULL, 10 /* uBase */, &pProxyDevUsbIp->uPort);
698 if ( rc == VINF_SUCCESS
699 || rc == VWRN_TRAILING_CHARS)
700 {
701 rc = RTStrAllocEx(&pProxyDevUsbIp->pszHost, cbHost + 1);
702 if (RT_SUCCESS(rc))
703 rc = RTStrAllocEx(&pProxyDevUsbIp->pszBusId, cbBusId + 1);
704 if (RT_SUCCESS(rc))
705 {
706 rc = RTStrCopyEx(pProxyDevUsbIp->pszHost, cbHost + 1, pszAddress, cbHost);
707 AssertRC(rc);
708
709 rc = RTStrCopyEx(pProxyDevUsbIp->pszBusId, cbBusId + 1, pszBusIdStart, cbBusId);
710 AssertRC(rc);
711
712 return VINF_SUCCESS;
713 }
714 }
715 else
716 rc = VERR_INVALID_PARAMETER;
717 }
718 else
719 rc = VERR_INVALID_PARAMETER;
720 }
721 else
722 rc = VERR_INVALID_PARAMETER;
723 }
724 else
725 rc = VERR_INVALID_PARAMETER;
726
727 return rc;
728}
729
730/**
731 * Connects to the USB/IP host and claims the device given in the proxy device data.
732 *
733 * @returns VBox status code.
734 * @param pProxyDevUsbIp The USB/IP proxy device data.
735 */
736static int usbProxyUsbIpConnect(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
737{
738 int rc = VINF_SUCCESS;
739 rc = RTTcpClientConnect(pProxyDevUsbIp->pszHost, pProxyDevUsbIp->uPort, &pProxyDevUsbIp->hSocket);
740 if (RT_SUCCESS(rc))
741 {
742 /* Disable send coalescing. */
743 rc = RTTcpSetSendCoalescing(pProxyDevUsbIp->hSocket, false);
744 if (RT_FAILURE(rc))
745 LogRel(("UsbIp: Disabling send coalescing failed (rc=%Rrc), continuing nevertheless but expect reduced performance\n", rc));
746
747 /* Import the device, i.e. claim it for our use. */
748 UsbIpReqImport ReqImport;
749 ReqImport.u16Version = RT_H2N_U16(USBIP_VERSION);
750 ReqImport.u16Cmd = RT_H2N_U16(USBIP_INDICATOR_REQ | USBIP_REQ_RET_IMPORT);
751 ReqImport.u32Status = RT_H2N_U32(USBIP_STATUS_SUCCESS);
752 rc = RTStrCopy(&ReqImport.aszBusId[0], sizeof(ReqImport.aszBusId), pProxyDevUsbIp->pszBusId);
753 if (rc == VINF_SUCCESS)
754 {
755 rc = RTTcpWrite(pProxyDevUsbIp->hSocket, &ReqImport, sizeof(ReqImport));
756 if (RT_SUCCESS(rc))
757 {
758 /* Read the reply. */
759 UsbIpRetImport RetImport;
760 rc = RTTcpRead(pProxyDevUsbIp->hSocket, &RetImport, sizeof(RetImport), NULL);
761 if (RT_SUCCESS(rc))
762 {
763 RetImport.u16Version = RT_N2H_U16(RetImport.u16Version);
764 RetImport.u16Cmd = RT_N2H_U16(RetImport.u16Cmd);
765 RetImport.u32Status = RT_N2H_U32(RetImport.u32Status);
766 if ( RetImport.u16Version == USBIP_VERSION
767 && RetImport.u16Cmd == USBIP_REQ_RET_IMPORT
768 && RetImport.u32Status == USBIP_STATUS_SUCCESS)
769 {
770 /* Read the device data. */
771 UsbIpExportedDevice Device;
772 rc = RTTcpRead(pProxyDevUsbIp->hSocket, &Device, sizeof(Device), NULL);
773 if (RT_SUCCESS(rc))
774 {
775 usbProxyUsbIpExportedDeviceN2H(&Device);
776 pProxyDevUsbIp->u32DevId = (Device.u32BusNum << 16) | Device.u32DevNum;
777
778 rc = RTPollSetAddSocket(pProxyDevUsbIp->hPollSet, pProxyDevUsbIp->hSocket,
779 RTPOLL_EVT_READ | RTPOLL_EVT_ERROR, USBIP_POLL_ID_SOCKET);
780 }
781 }
782 else
783 {
784 /* Check what went wrong and leave a meaningful error message in the log. */
785 if (RetImport.u16Version != USBIP_VERSION)
786 LogRel(("UsbIp: Unexpected protocol version received from host (%#x vs. %#x)\n",
787 RetImport.u16Version, USBIP_VERSION));
788 else if (RetImport.u16Cmd != USBIP_REQ_RET_IMPORT)
789 LogRel(("UsbIp: Unexpected reply code received from host (%#x vs. %#x)\n",
790 RetImport.u16Cmd, USBIP_REQ_RET_IMPORT));
791 else if (RetImport.u32Status != 0)
792 LogRel(("UsbIp: Claiming the device has failed on the host with an unspecified error\n"));
793 else
794 AssertMsgFailed(("Something went wrong with if condition\n"));
795 }
796 }
797 }
798 }
799 else
800 {
801 LogRel(("UsbIp: Given bus ID is exceeds permitted protocol length: %u vs %u\n",
802 strlen(pProxyDevUsbIp->pszBusId) + 1, sizeof(ReqImport.aszBusId)));
803 rc = VERR_INVALID_PARAMETER;
804 }
805
806 if (RT_FAILURE(rc))
807 RTTcpClientCloseEx(pProxyDevUsbIp->hSocket, false /*fGracefulShutdown*/);
808 }
809 if (RT_FAILURE(rc))
810 LogRel(("UsbIp: Connecting to the host %s failed with %Rrc\n", pProxyDevUsbIp->pszHost, rc));
811 return rc;
812}
813
814/**
815 * Disconnects from the USB/IP host releasing the device given in the proxy device data.
816 *
817 * @returns VBox status code.
818 * @param pProxyDevUsbIp The USB/IP proxy device data.
819 */
820static int usbProxyUsbIpDisconnect(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
821{
822 int rc = VINF_SUCCESS;
823
824 rc = RTTcpClientCloseEx(pProxyDevUsbIp->hSocket, false /*fGracefulShutdown*/);
825 if (RT_SUCCESS(rc))
826 pProxyDevUsbIp->hSocket = NIL_RTSOCKET;
827 return rc;
828}
829
830/**
831 * Synchronously exchange a given control message with the remote device.
832 *
833 * @eturns VBox status code.
834 * @param pProxyDevUsbIp The USB/IP proxy device data.
835 * @param pSetup The setup message.
836 *
837 * @note This method is only used to implement the *SetConfig, *SetInterface and *ClearHaltedEp
838 * callbacks because the USB/IP protocol lacks dedicated requests for these.
839 * @remark It is assumed that this method is never called while usbProxyUsbIpUrbReap is called
840 * on another thread.
841 */
842static int usbProxyUsbIpCtrlUrbExchangeSync(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PVUSBSETUP pSetup)
843{
844 int rc = VINF_SUCCESS;
845
846 UsbIpReqSubmit ReqSubmit;
847 uint32_t u32SeqNum = usbProxyUsbIpSeqNumGet(pProxyDevUsbIp);
848 ReqSubmit.Hdr.u32ReqRet = USBIP_CMD_SUBMIT;
849 ReqSubmit.Hdr.u32SeqNum = u32SeqNum;
850 ReqSubmit.Hdr.u32DevId = pProxyDevUsbIp->u32DevId;
851 ReqSubmit.Hdr.u32Direction = USBIP_DIR_OUT;
852 ReqSubmit.Hdr.u32Endpoint = 0; /* Only default control endpoint is allowed for these kind of messages. */
853 ReqSubmit.u32XferFlags = 0;
854 ReqSubmit.u32TransferBufferLength = 0;
855 ReqSubmit.u32StartFrame = 0;
856 ReqSubmit.u32NumIsocPkts = 0;
857 ReqSubmit.u32Interval = 0;
858 memcpy(&ReqSubmit.Setup, pSetup, sizeof(ReqSubmit.Setup));
859 usbProxyUsbIpReqSubmitH2N(&ReqSubmit);
860
861 /* Send the command. */
862 rc = RTTcpWrite(pProxyDevUsbIp->hSocket, &ReqSubmit, sizeof(ReqSubmit));
863 if (RT_SUCCESS(rc))
864 {
865 /* Wait for the response. */
866 /** @todo: Don't wait indefinitely long. */
867 UsbIpRetSubmit RetSubmit;
868 rc = RTTcpRead(pProxyDevUsbIp->hSocket, &RetSubmit, sizeof(RetSubmit), NULL);
869 if (RT_SUCCESS(rc))
870 {
871 usbProxyUsbIpRetSubmitN2H(&RetSubmit);
872 rc = usbProxyUsbIpStatusConvertFromStatus(RetSubmit.u32Status);
873 }
874 }
875 return rc;
876}
877
878/**
879 * Returns the URB matching the given sequence number from the in flight list.
880 *
881 * @returns pointer to the URB matching the given sequence number or NULL
882 * @param pProxyDevUsbIp The USB/IP proxy device data.
883 * @param u32SeqNum The sequence number to search for.
884 */
885static PUSBPROXYURBUSBIP usbProxyUsbIpGetUrbFromSeqNum(PUSBPROXYDEVUSBIP pProxyDevUsbIp, uint32_t u32SeqNum)
886{
887 bool fFound = false;
888 PUSBPROXYURBUSBIP pIt;
889
890 RTListForEach(&pProxyDevUsbIp->ListUrbsInFlight, pIt, USBPROXYURBUSBIP, NodeList)
891 {
892 if (pIt->u32SeqNumUrb == u32SeqNum)
893 {
894 fFound = true;
895 break;
896 }
897 }
898
899 return fFound ? pIt : NULL;
900}
901
902/**
903 * Resets the receive state for a new reply.
904 *
905 * @returns nothing.
906 * @param pProxyDevUsbIp The USB/IP proxy device data.
907 */
908static void usbProxyUsbIpResetRecvState(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
909{
910 pProxyDevUsbIp->enmRecvState = USBPROXYUSBIPRECVSTATE_HDR_COMMON;
911 pProxyDevUsbIp->pbRecv = (uint8_t *)&pProxyDevUsbIp->BufRet;
912 pProxyDevUsbIp->cbRecv = 0;
913 pProxyDevUsbIp->cbLeft = sizeof(UsbIpReqRetHdr);
914}
915
916static void usbProxyUsbIpRecvStateAdvance(PUSBPROXYDEVUSBIP pProxyDevUsbIp, USBPROXYUSBIPRECVSTATE enmState,
917 uint8_t *pbData, size_t cbData)
918{
919 pProxyDevUsbIp->enmRecvState = enmState;
920 pProxyDevUsbIp->cbRecv = 0;
921 pProxyDevUsbIp->cbLeft = cbData;
922 pProxyDevUsbIp->pbRecv = pbData;
923}
924
925/**
926 * Handles reception of a USB/IP PDU.
927 *
928 * @returns VBox status code.
929 * @param pProxyDevUsbIp The USB/IP proxy device data.
930 * @param ppUrbUsbIp Where to store the pointer to the USB/IP URB which completed.
931 * Will be NULL if the received PDU is not complete and we have
932 * have to wait for more data or on failure.
933 */
934static int usbProxyUsbIpRecvPdu(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PUSBPROXYURBUSBIP *ppUrbUsbIp)
935{
936 int rc = VINF_SUCCESS;
937 size_t cbRead = 0;
938 PUSBPROXYURBUSBIP pUrbUsbIp = NULL;
939
940 Assert(pProxyDevUsbIp->cbLeft);
941
942 /* Read any available data first. */
943 rc = RTTcpReadNB(pProxyDevUsbIp->hSocket, pProxyDevUsbIp->pbRecv, pProxyDevUsbIp->cbLeft, &cbRead);
944 if (RT_SUCCESS(rc))
945 {
946 pProxyDevUsbIp->cbRecv += cbRead;
947 pProxyDevUsbIp->cbLeft -= cbRead;
948 pProxyDevUsbIp->pbRecv += cbRead;
949
950 /* Process the received data if there is nothing to receive left for the current state. */
951 if (!pProxyDevUsbIp->cbLeft)
952 {
953 switch (pProxyDevUsbIp->enmRecvState)
954 {
955 case USBPROXYUSBIPRECVSTATE_HDR_COMMON:
956 {
957 Assert(pProxyDevUsbIp->cbRecv == sizeof(UsbIpReqRetHdr));
958
959 /*
960 * Determine the residual amount of data to receive until
961 * the complete reply header was received.
962 */
963 switch (RT_N2H_U32(pProxyDevUsbIp->BufRet.Hdr.u32ReqRet))
964 {
965 case USBIP_RET_SUBMIT:
966 pProxyDevUsbIp->cbLeft = sizeof(UsbIpRetSubmit) - sizeof(UsbIpReqRetHdr);
967 pProxyDevUsbIp->enmRecvState = USBPROXYUSBIPRECVSTATE_HDR_RESIDUAL;
968 break;
969 case USBIP_RET_UNLINK:
970 pProxyDevUsbIp->cbLeft = sizeof(UsbIpRetUnlink) - sizeof(UsbIpReqRetHdr);
971 pProxyDevUsbIp->enmRecvState = USBPROXYUSBIPRECVSTATE_HDR_RESIDUAL;
972 break;
973 default:
974 AssertLogRelMsgFailed(("Invalid reply header received: %d\n",
975 pProxyDevUsbIp->BufRet.Hdr.u32ReqRet));
976 usbProxyUsbIpResetRecvState(pProxyDevUsbIp);
977 }
978
979 break;
980 }
981 case USBPROXYUSBIPRECVSTATE_HDR_RESIDUAL:
982 {
983 /* Get the URB from the in flight list. */
984 pProxyDevUsbIp->pUrbUsbIp = usbProxyUsbIpGetUrbFromSeqNum(pProxyDevUsbIp, RT_N2H_U32(pProxyDevUsbIp->BufRet.Hdr.u32SeqNum));
985 if (pProxyDevUsbIp->pUrbUsbIp)
986 {
987 /** @todo: Verify that the directions match, verify that the length doesn't exceed the buffer. */
988
989 switch (RT_N2H_U32(pProxyDevUsbIp->BufRet.Hdr.u32ReqRet))
990 {
991 case USBIP_RET_SUBMIT:
992 usbProxyUsbIpRetSubmitN2H(&pProxyDevUsbIp->BufRet.RetSubmit);
993
994 pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->enmStatus = usbProxyUsbIpVUsbStatusConvertFromStatus(pProxyDevUsbIp->BufRet.RetSubmit.u32Status);
995 if ( pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->enmDir == VUSBDIRECTION_IN
996 && pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->enmStatus == VUSBSTATUS_OK)
997 {
998 uint8_t *pbData = NULL;
999
1000 if (pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->enmType == VUSBXFERTYPE_MSG)
1001 {
1002 /* Preserve the setup request. */
1003 pbData = &pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->abData[sizeof(VUSBSETUP)];
1004 pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->cbData = pProxyDevUsbIp->BufRet.RetSubmit.u32ActualLength + sizeof(VUSBSETUP);
1005 }
1006 else
1007 {
1008 pbData = &pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->abData[0];
1009 pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->cbData = pProxyDevUsbIp->BufRet.RetSubmit.u32ActualLength;
1010 }
1011
1012 usbProxyUsbIpRecvStateAdvance(pProxyDevUsbIp, USBPROXYUSBIPRECVSTATE_URB_BUFFER,
1013 pbData, pProxyDevUsbIp->BufRet.RetSubmit.u32ActualLength);
1014 }
1015 else
1016 {
1017 Assert( pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->enmDir == VUSBDIRECTION_OUT
1018 || pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->enmStatus != VUSBSTATUS_OK);
1019 pUrbUsbIp = pProxyDevUsbIp->pUrbUsbIp;
1020 usbProxyUsbIpResetRecvState(pProxyDevUsbIp);
1021 }
1022 break;
1023 case USBIP_RET_UNLINK:
1024 usbProxyUsbIpRetUnlinkN2H(&pProxyDevUsbIp->BufRet.RetUnlink);
1025 pUrbUsbIp = pProxyDevUsbIp->pUrbUsbIp;
1026 pUrbUsbIp->pVUsbUrb->enmStatus = usbProxyUsbIpVUsbStatusConvertFromStatus(pProxyDevUsbIp->BufRet.RetUnlink.u32Status);
1027 usbProxyUsbIpResetRecvState(pProxyDevUsbIp);
1028 break;
1029 }
1030 }
1031 else
1032 {
1033 LogRel(("USB/IP: Received reply with sequence number doesn't match any local URB\n", pProxyDevUsbIp->BufRet.Hdr.u32SeqNum));
1034 usbProxyUsbIpResetRecvState(pProxyDevUsbIp);
1035 }
1036
1037 break;
1038 }
1039 case USBPROXYUSBIPRECVSTATE_URB_BUFFER:
1040 if (pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->enmType == VUSBXFERTYPE_ISOC)
1041 usbProxyUsbIpRecvStateAdvance(pProxyDevUsbIp, USBPROXYUSBIPRECVSTATE_ISOC_PKT_DESCS,
1042 (uint8_t *)&pProxyDevUsbIp->aIsocPktDesc[0],
1043 pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->cIsocPkts * sizeof(UsbIpIsocPktDesc));
1044 else
1045 {
1046 pUrbUsbIp = pProxyDevUsbIp->pUrbUsbIp;
1047 usbProxyUsbIpResetRecvState(pProxyDevUsbIp);
1048 }
1049 break;
1050 case USBPROXYUSBIPRECVSTATE_ISOC_PKT_DESCS:
1051 /* Process all received isochronous packet descriptors. */
1052 for (unsigned i = 0; i < pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->cIsocPkts; i++)
1053 {
1054 PVUSBURBISOCPTK pIsocPkt = &pProxyDevUsbIp->pUrbUsbIp->pVUsbUrb->aIsocPkts[i];
1055 usbProxyUsbIpIsocPktDescN2H(&pProxyDevUsbIp->aIsocPktDesc[i]);
1056 pIsocPkt->enmStatus = usbProxyUsbIpVUsbStatusConvertFromStatus(pProxyDevUsbIp->aIsocPktDesc[i].i32Status);
1057 pIsocPkt->off = pProxyDevUsbIp->aIsocPktDesc[i].u32Offset;
1058 pIsocPkt->cb = pProxyDevUsbIp->aIsocPktDesc[i].u32ActualLength;
1059 }
1060
1061 pUrbUsbIp = pProxyDevUsbIp->pUrbUsbIp;
1062 usbProxyUsbIpResetRecvState(pProxyDevUsbIp);
1063 break;
1064 default:
1065 AssertLogRelMsgFailed(("USB/IP: Invalid receive state %d\n", pProxyDevUsbIp->enmRecvState));
1066 }
1067 }
1068 }
1069 else
1070 {
1071 /** @todo: Complete all URBs with DNR error and mark device as unplugged. */
1072#if 0
1073 pUrbUsbIp = pProxyDevUsbIp->pUrbUsbIp;
1074 pUrbUsbIp->pVUsbUrb->enmStatus = VUSBSTATUS_DNR;
1075 usbProxyUsbIpResetRecvState(pProxyDevUsbIp);
1076#endif
1077 }
1078
1079 if (RT_SUCCESS(rc))
1080 *ppUrbUsbIp = pUrbUsbIp;
1081
1082 return rc;
1083}
1084
1085/**
1086 * Worker for queueing an URB on the main I/O thread.
1087 *
1088 * @returns VBox status code.
1089 * @param pProxyDevUsbIp The USB/IP proxy device data.
1090 * @param pUrbUsbIp The USB/IP URB to queue.
1091 */
1092static int usbProxyUsbIpUrbQueueWorker(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PUSBPROXYURBUSBIP pUrbUsbIp)
1093{
1094 PVUSBURB pUrb = pUrbUsbIp->pVUsbUrb;
1095
1096 pUrbUsbIp->u32SeqNumUrb = usbProxyUsbIpSeqNumGet(pProxyDevUsbIp);
1097
1098 UsbIpReqSubmit ReqSubmit;
1099
1100 RT_ZERO(ReqSubmit);
1101 ReqSubmit.Hdr.u32ReqRet = USBIP_CMD_SUBMIT;
1102 ReqSubmit.Hdr.u32SeqNum = pUrbUsbIp->u32SeqNumUrb;
1103 ReqSubmit.Hdr.u32DevId = pProxyDevUsbIp->u32DevId;
1104 ReqSubmit.Hdr.u32Endpoint = pUrb->EndPt;
1105 ReqSubmit.Hdr.u32Direction = pUrb->enmDir == VUSBDIRECTION_IN ? USBIP_DIR_IN : USBIP_DIR_OUT;
1106 ReqSubmit.u32XferFlags = 0;
1107 if (pUrb->enmDir == VUSBDIRECTION_IN && pUrb->fShortNotOk)
1108 ReqSubmit.u32XferFlags |= USBIP_XFER_FLAGS_SHORT_NOT_OK;
1109
1110 ReqSubmit.u32TransferBufferLength = pUrb->cbData;
1111 ReqSubmit.u32StartFrame = 0;
1112 ReqSubmit.u32NumIsocPkts = 0;
1113 ReqSubmit.u32Interval = 0;
1114
1115 RTSGSEG aSegReq[3]; /* Maximum number of segments used for a Isochronous transfer. */
1116 UsbIpIsocPktDesc aIsocPktsDesc[8];
1117 unsigned cSegsUsed = 1;
1118 aSegReq[0].pvSeg = &ReqSubmit;
1119 aSegReq[0].cbSeg = sizeof(ReqSubmit);
1120
1121
1122 switch (pUrb->enmType)
1123 {
1124 case VUSBXFERTYPE_MSG:
1125 memcpy(&ReqSubmit.Setup, &pUrb->abData, sizeof(ReqSubmit.Setup));
1126 if (pUrb->enmDir == VUSBDIRECTION_OUT)
1127 {
1128 aSegReq[cSegsUsed].cbSeg = pUrb->cbData - sizeof(VUSBSETUP);
1129 aSegReq[cSegsUsed].pvSeg = pUrb->abData + sizeof(VUSBSETUP);
1130 cSegsUsed++;
1131 }
1132 LogFlowFunc(("Message (Control) URB\n"));
1133 break;
1134 case VUSBXFERTYPE_ISOC:
1135 ReqSubmit.u32XferFlags |= USBIP_XFER_FLAGS_ISO_ASAP;
1136 ReqSubmit.u32NumIsocPkts = pUrb->cIsocPkts;
1137 if (pUrb->enmDir == VUSBDIRECTION_OUT)
1138 {
1139 aSegReq[cSegsUsed].cbSeg = pUrb->cbData;
1140 aSegReq[cSegsUsed].pvSeg = pUrb->abData;
1141 cSegsUsed++;
1142 }
1143
1144 for (unsigned i = 0; i < pUrb->cIsocPkts; i++)
1145 {
1146 aIsocPktsDesc[i].u32Offset = pUrb->aIsocPkts[i].off;
1147 aIsocPktsDesc[i].u32Length = pUrb->aIsocPkts[i].cb;
1148 aIsocPktsDesc[i].u32ActualLength = 0; /** @todo */
1149 aIsocPktsDesc[i].i32Status = pUrb->aIsocPkts[i].enmStatus;
1150 usbProxyUsbIpIsocPktDescH2N(&aIsocPktsDesc[i]);
1151 }
1152
1153 if (pUrb->cIsocPkts)
1154 {
1155 aSegReq[cSegsUsed].cbSeg = pUrb->cIsocPkts * sizeof(UsbIpIsocPktDesc);
1156 aSegReq[cSegsUsed].pvSeg = &aIsocPktsDesc[0];
1157 cSegsUsed++;
1158 }
1159
1160 break;
1161 case VUSBXFERTYPE_BULK:
1162 case VUSBXFERTYPE_INTR:
1163 if (pUrb->enmDir == VUSBDIRECTION_OUT)
1164 {
1165 aSegReq[cSegsUsed].cbSeg = pUrb->cbData;
1166 aSegReq[cSegsUsed].pvSeg = pUrb->abData;
1167 cSegsUsed++;
1168 }
1169 break;
1170 default:
1171 usbProxyUsbIpUrbFree(pProxyDevUsbIp, pUrbUsbIp);
1172 return VERR_INVALID_PARAMETER; /** @todo: better status code. */
1173 }
1174
1175 usbProxyUsbIpReqSubmitH2N(&ReqSubmit);
1176
1177 Assert(cSegsUsed <= RT_ELEMENTS(aSegReq));
1178
1179 /* Send the command. */
1180 RTSGBUF SgBufReq;
1181 RTSgBufInit(&SgBufReq, &aSegReq[0], cSegsUsed);
1182
1183 int rc = RTTcpSgWrite(pProxyDevUsbIp->hSocket, &SgBufReq);
1184 if (RT_SUCCESS(rc))
1185 {
1186 /* Link the URB into the list of in flight URBs. */
1187 usbProxyUsbIpLinkUrb(pProxyDevUsbIp, &pProxyDevUsbIp->ListUrbsInFlight, pUrbUsbIp);
1188 }
1189
1190 return rc;
1191}
1192
1193/**
1194 * Queues all pending URBs from the list.
1195 *
1196 * @returns VBox status code.
1197 * @param pProxyDevUsbIp The USB/IP proxy device data.
1198 */
1199static int usbProxyUsbIpUrbsQueuePending(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
1200{
1201 RTLISTANCHOR ListUrbsPending;
1202
1203 RTListInit(&ListUrbsPending);
1204 int rc = RTSemFastMutexRequest(pProxyDevUsbIp->hMtxLists);
1205 AssertRC(rc);
1206 RTListMove(&ListUrbsPending, &pProxyDevUsbIp->ListUrbsToQueue);
1207 RTSemFastMutexRelease(pProxyDevUsbIp->hMtxLists);
1208
1209 PUSBPROXYURBUSBIP pIter = NULL;
1210 PUSBPROXYURBUSBIP pIterNext = NULL;
1211 RTListForEachSafe(&ListUrbsPending, pIter, pIterNext, USBPROXYURBUSBIP, NodeList)
1212 {
1213 RTListNodeRemove(&pIter->NodeList);
1214 rc = usbProxyUsbIpUrbQueueWorker(pProxyDevUsbIp, pIter);
1215 if (RT_FAILURE(rc))
1216 {
1217 /** @todo: Complete the URB with an error. */
1218 usbProxyUsbIpUrbFree(pProxyDevUsbIp, pIter);
1219 }
1220 }
1221
1222 return VINF_SUCCESS;
1223}
1224
1225/**
1226 * Kick the reaper thread.
1227 *
1228 * @returns VBox status code.
1229 * @param pProxyDevUsbIp The USB/IP proxy device data.
1230 * @param bReason The wakeup reason.
1231 */
1232static char usbProxyReaperKick(PUSBPROXYDEVUSBIP pProxyDevUsbIp, char bReason)
1233{
1234 int rc = VINF_SUCCESS;
1235 size_t cbWritten = 0;
1236
1237 rc = RTPipeWrite(pProxyDevUsbIp->hPipeW, &bReason, 1, &cbWritten);
1238 Assert(RT_SUCCESS(rc) || cbWritten == 0);
1239
1240 return rc;
1241}
1242
1243/**
1244 * Drain the wakeup pipe.
1245 *
1246 * @returns Wakeup reason.
1247 * @param pProxyDevUsbIp The USB/IP proxy device data.
1248 */
1249static char usbProxyUsbIpWakeupPipeDrain(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
1250{
1251 char bRead = 0;
1252 size_t cbRead = 0;
1253
1254 int rc = RTPipeRead(pProxyDevUsbIp->hPipeR, &bRead, 1, &cbRead);
1255 Assert(RT_SUCCESS(rc) && cbRead == 1);
1256
1257 return bRead;
1258}
1259
1260/*
1261 * The USB proxy device functions.
1262 */
1263
1264static DECLCALLBACK(int) usbProxyUsbIpOpen(PUSBPROXYDEV pProxyDev, const char *pszAddress, void *pvBackend)
1265{
1266 LogFlowFunc(("pProxyDev=%p pszAddress=%s, pvBackend=%p\n", pProxyDev, pszAddress, pvBackend));
1267
1268 PUSBPROXYDEVUSBIP pDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1269 int rc = VINF_SUCCESS;
1270
1271 RTListInit(&pDevUsbIp->ListUrbsInFlight);
1272 RTListInit(&pDevUsbIp->ListUrbsLanded);
1273 RTListInit(&pDevUsbIp->ListUrbsToQueue);
1274 pDevUsbIp->hSocket = NIL_RTSOCKET;
1275 pDevUsbIp->hPollSet = NIL_RTPOLLSET;
1276 pDevUsbIp->hPipeW = NIL_RTPIPE;
1277 pDevUsbIp->hPipeR = NIL_RTPIPE;
1278 pDevUsbIp->u32SeqNumNext = 0;
1279 pDevUsbIp->pszHost = NULL;
1280 pDevUsbIp->pszBusId = NULL;
1281 usbProxyUsbIpResetRecvState(pDevUsbIp);
1282
1283 rc = RTSemFastMutexCreate(&pDevUsbIp->hMtxLists);
1284 if (RT_SUCCESS(rc))
1285 {
1286 /* Setup wakeup pipe and poll set first. */
1287 rc = RTPipeCreate(&pDevUsbIp->hPipeR, &pDevUsbIp->hPipeW, 0);
1288 if (RT_SUCCESS(rc))
1289 {
1290 rc = RTPollSetCreate(&pDevUsbIp->hPollSet);
1291 if (RT_SUCCESS(rc))
1292 {
1293 rc = RTPollSetAddPipe(pDevUsbIp->hPollSet, pDevUsbIp->hPipeR,
1294 RTPOLL_EVT_READ, USBIP_POLL_ID_PIPE);
1295 if (RT_SUCCESS(rc))
1296 {
1297 /* Connect to the USB/IP host. */
1298 rc = usbProxyUsbIpParseAddress(pDevUsbIp, pszAddress);
1299 if (RT_SUCCESS(rc))
1300 rc = usbProxyUsbIpConnect(pDevUsbIp);
1301 }
1302
1303 if (RT_FAILURE(rc))
1304 {
1305 RTPollSetRemove(pDevUsbIp->hPollSet, USBIP_POLL_ID_PIPE);
1306 int rc2 = RTPollSetDestroy(pDevUsbIp->hPollSet);
1307 AssertRC(rc2);
1308 }
1309 }
1310
1311 if (RT_FAILURE(rc))
1312 {
1313 int rc2 = RTPipeClose(pDevUsbIp->hPipeR);
1314 AssertRC(rc2);
1315 rc2 = RTPipeClose(pDevUsbIp->hPipeW);
1316 AssertRC(rc2);
1317 }
1318 }
1319 }
1320
1321 return rc;
1322}
1323
1324static DECLCALLBACK(void) usbProxyUsbIpClose(PUSBPROXYDEV pProxyDev)
1325{
1326 int rc = VINF_SUCCESS;
1327 LogFlowFunc(("pProxyDev = %p\n", pProxyDev));
1328
1329 PUSBPROXYDEVUSBIP pDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1330 /* Destroy the pipe and pollset if necessary. */
1331 if (pDevUsbIp->hPollSet != NIL_RTPOLLSET)
1332 {
1333 if (pDevUsbIp->hSocket != NIL_RTSOCKET)
1334 {
1335 rc = RTPollSetRemove(pDevUsbIp->hPollSet, USBIP_POLL_ID_SOCKET);
1336 Assert(RT_SUCCESS(rc) || rc == VERR_POLL_HANDLE_ID_NOT_FOUND);
1337 }
1338 rc = RTPollSetRemove(pDevUsbIp->hPollSet, USBIP_POLL_ID_PIPE);
1339 AssertRC(rc);
1340 rc = RTPollSetDestroy(pDevUsbIp->hPollSet);
1341 AssertRC(rc);
1342 rc = RTPipeClose(pDevUsbIp->hPipeR);
1343 AssertRC(rc);
1344 rc = RTPipeClose(pDevUsbIp->hPipeW);
1345 AssertRC(rc);
1346 }
1347
1348 if (pDevUsbIp->hSocket != NIL_RTSOCKET)
1349 usbProxyUsbIpDisconnect(pDevUsbIp);
1350 if (pDevUsbIp->pszHost)
1351 RTStrFree(pDevUsbIp->pszHost);
1352 if (pDevUsbIp->pszBusId)
1353 RTStrFree(pDevUsbIp->pszBusId);
1354
1355 /* Clear the URB lists. */
1356 rc = RTSemFastMutexRequest(pDevUsbIp->hMtxLists);
1357 AssertRC(rc);
1358 PUSBPROXYURBUSBIP pIter = NULL;
1359 PUSBPROXYURBUSBIP pIterNext = NULL;
1360 RTListForEachSafe(&pDevUsbIp->ListUrbsInFlight, pIter, pIterNext, USBPROXYURBUSBIP, NodeList)
1361 {
1362 RTListNodeRemove(&pIter->NodeList);
1363 RTMemFree(pIter);
1364 }
1365
1366 RTListForEachSafe(&pDevUsbIp->ListUrbsLanded, pIter, pIterNext, USBPROXYURBUSBIP, NodeList)
1367 {
1368 RTListNodeRemove(&pIter->NodeList);
1369 RTMemFree(pIter);
1370 }
1371 RTSemFastMutexRelease(pDevUsbIp->hMtxLists);
1372 RTSemFastMutexDestroy(pDevUsbIp->hMtxLists);
1373}
1374
1375static DECLCALLBACK(int) usbProxyUsbIpReset(PUSBPROXYDEV pProxyDev, bool fResetOnLinux)
1376{
1377 LogFlowFunc(("pProxyDev = %p\n", pProxyDev));
1378
1379 int rc = VINF_SUCCESS;
1380 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1381 VUSBSETUP Setup;
1382
1383 if (fResetOnLinux)
1384 {
1385 Setup.bmRequestType = RT_BIT(5) | 0x03; /* Port request. */
1386 Setup.bRequest = 0x03; /* SET_FEATURE */
1387 Setup.wValue = 4; /* Port feature: Reset */
1388 Setup.wIndex = 0; /* Port number, irrelevant */
1389 Setup.wLength = 0;
1390 rc = usbProxyUsbIpCtrlUrbExchangeSync(pProxyDevUsbIp, &Setup);
1391 if (RT_SUCCESS(rc))
1392 {
1393 pProxyDev->iActiveCfg = -1;
1394 pProxyDev->cIgnoreSetConfigs = 2;
1395 }
1396 }
1397
1398 return rc;
1399}
1400
1401static DECLCALLBACK(int) usbProxyUsbIpSetConfig(PUSBPROXYDEV pProxyDev, int iCfg)
1402{
1403 LogFlowFunc(("pProxyDev=%s cfg=%#x\n", pProxyDev->pUsbIns->pszName, iCfg));
1404
1405 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1406 VUSBSETUP Setup;
1407
1408 Setup.bmRequestType = 0;
1409 Setup.bRequest = 0x09;
1410 Setup.wValue = iCfg;
1411 Setup.wIndex = 0;
1412 Setup.wLength = 0;
1413 return usbProxyUsbIpCtrlUrbExchangeSync(pProxyDevUsbIp, &Setup);
1414}
1415
1416static DECLCALLBACK(int) usbProxyUsbIpClaimInterface(PUSBPROXYDEV pProxyDev, int ifnum)
1417{
1418 LogFlowFunc(("pProxyDev=%s ifnum=%#x\n", pProxyDev->pUsbIns->pszName, ifnum));
1419 return VINF_SUCCESS;
1420}
1421
1422static DECLCALLBACK(int) usbProxyUsbIpReleaseInterface(PUSBPROXYDEV pProxyDev, int ifnum)
1423{
1424 LogFlowFunc(("pProxyDev=%s ifnum=%#x\n", pProxyDev->pUsbIns->pszName, ifnum));
1425 return VINF_SUCCESS;
1426}
1427
1428static DECLCALLBACK(int) usbProxyUsbIpSetInterface(PUSBPROXYDEV pProxyDev, int ifnum, int setting)
1429{
1430 LogFlowFunc(("pProxyDev=%p ifnum=%#x setting=%#x\n", pProxyDev, ifnum, setting));
1431
1432 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1433 VUSBSETUP Setup;
1434
1435 Setup.bmRequestType = 0x1;
1436 Setup.bRequest = 0x0b; /* SET_INTERFACE */
1437 Setup.wValue = setting;
1438 Setup.wIndex = ifnum;
1439 Setup.wLength = 0;
1440 return usbProxyUsbIpCtrlUrbExchangeSync(pProxyDevUsbIp, &Setup);
1441}
1442
1443static DECLCALLBACK(int) usbProxyUsbIpClearHaltedEp(PUSBPROXYDEV pProxyDev, unsigned int iEp)
1444{
1445 LogFlowFunc(("pProxyDev=%s ep=%u\n", pProxyDev->pUsbIns->pszName, iEp));
1446
1447 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1448 VUSBSETUP Setup;
1449
1450 Setup.bmRequestType = 0x2;
1451 Setup.bRequest = 0x01; /* CLEAR_FEATURE */
1452 Setup.wValue = 0x00; /* ENDPOINT_HALT */
1453 Setup.wIndex = iEp;
1454 Setup.wLength = 0;
1455 return usbProxyUsbIpCtrlUrbExchangeSync(pProxyDevUsbIp, &Setup);
1456}
1457
1458static DECLCALLBACK(int) usbProxyUsbIpUrbQueue(PUSBPROXYDEV pProxyDev, PVUSBURB pUrb)
1459{
1460 LogFlowFunc(("pUrb=%p\n", pUrb));
1461
1462 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1463
1464 /* Allocate a USB/IP Urb. */
1465 PUSBPROXYURBUSBIP pUrbUsbIp = usbProxyUsbIpUrbAlloc(pProxyDevUsbIp);
1466 if (!pUrbUsbIp)
1467 return VERR_NO_MEMORY;
1468
1469 pUrbUsbIp->pVUsbUrb = pUrb;
1470 pUrb->Dev.pvPrivate = pUrbUsbIp;
1471
1472 int rc = RTSemFastMutexRequest(pProxyDevUsbIp->hMtxLists);
1473 AssertRC(rc);
1474 RTListAppend(&pProxyDevUsbIp->ListUrbsToQueue, &pUrbUsbIp->NodeList);
1475 RTSemFastMutexRelease(pProxyDevUsbIp->hMtxLists);
1476
1477 return usbProxyReaperKick(pProxyDevUsbIp, USBIP_REAPER_WAKEUP_REASON_QUEUE);
1478}
1479
1480static DECLCALLBACK(PVUSBURB) usbProxyUsbIpUrbReap(PUSBPROXYDEV pProxyDev, RTMSINTERVAL cMillies)
1481{
1482 LogFlowFunc(("pProxyDev=%s\n", pProxyDev->pUsbIns->pszName));
1483
1484 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1485 PUSBPROXYURBUSBIP pUrbUsbIp = NULL;
1486 PVUSBURB pUrb = NULL;
1487 int rc = VINF_SUCCESS;
1488
1489 /* Queue new URBs first. */
1490 rc = usbProxyUsbIpUrbsQueuePending(pProxyDevUsbIp);
1491 AssertRC(rc);
1492
1493 /* Any URBs pending delivery? */
1494 if (!RTListIsEmpty(&pProxyDevUsbIp->ListUrbsLanded))
1495 pUrbUsbIp = RTListGetFirst(&pProxyDevUsbIp->ListUrbsLanded, USBPROXYURBUSBIP, NodeList);
1496
1497 while (!pUrbUsbIp && RT_SUCCESS(rc) && cMillies)
1498 {
1499 uint32_t uIdReady = 0;
1500 uint32_t fEventsRecv = 0;
1501 RTMSINTERVAL msStart = RTTimeMilliTS();
1502 RTMSINTERVAL msNow;
1503
1504 rc = RTPoll(pProxyDevUsbIp->hPollSet, cMillies, &fEventsRecv, &uIdReady);
1505 Assert(RT_SUCCESS(rc) || rc == VERR_TIMEOUT);
1506 if (RT_SUCCESS(rc))
1507 {
1508 msNow = RTTimeMilliTS();
1509 cMillies = msNow - msStart >= cMillies ? 0 : cMillies - (msNow - msStart);
1510
1511 if (uIdReady == USBIP_POLL_ID_SOCKET)
1512 rc = usbProxyUsbIpRecvPdu(pProxyDevUsbIp, &pUrbUsbIp);
1513 else
1514 {
1515 AssertLogRelMsg(uIdReady == USBIP_POLL_ID_PIPE, ("Invalid pollset ID given\n"));
1516
1517 char bReason = usbProxyUsbIpWakeupPipeDrain(pProxyDevUsbIp);
1518 if (bReason == USBIP_REAPER_WAKEUP_REASON_QUEUE)
1519 usbProxyUsbIpUrbsQueuePending(pProxyDevUsbIp);
1520 else
1521 {
1522 Assert(bReason == USBIP_REAPER_WAKEUP_REASON_EXTERNAL);
1523 break;
1524 }
1525 }
1526 }
1527 }
1528
1529 if (pUrbUsbIp)
1530 {
1531 pUrb = pUrbUsbIp->pVUsbUrb;
1532
1533 /* unlink from the pending delivery list */
1534 usbProxyUsbIpUnlinkUrb(pProxyDevUsbIp, pUrbUsbIp);
1535 usbProxyUsbIpUrbFree(pProxyDevUsbIp, pUrbUsbIp);
1536 }
1537
1538 return pUrb;
1539}
1540
1541static DECLCALLBACK(int) usbProxyUsbIpUrbCancel(PUSBPROXYDEV pProxyDev, PVUSBURB pUrb)
1542{
1543 LogFlowFunc(("pUrb=%p\n", pUrb));
1544
1545 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1546 PUSBPROXYURBUSBIP pUrbUsbIp = (PUSBPROXYURBUSBIP)pUrb->Dev.pvPrivate;
1547 UsbIpReqUnlink ReqUnlink;
1548
1549 uint32_t u32SeqNum = usbProxyUsbIpSeqNumGet(pProxyDevUsbIp);
1550 ReqUnlink.Hdr.u32ReqRet = USBIP_CMD_UNLINK;
1551 ReqUnlink.Hdr.u32SeqNum = u32SeqNum;
1552 ReqUnlink.Hdr.u32DevId = pProxyDevUsbIp->u32DevId;
1553 ReqUnlink.Hdr.u32Direction = USBIP_DIR_OUT;
1554 ReqUnlink.Hdr.u32Endpoint = pUrb->EndPt;
1555 ReqUnlink.u32SeqNum = pUrbUsbIp->u32SeqNumUrb;
1556
1557 usbProxyUsbIpReqUnlinkH2N(&ReqUnlink);
1558 return RTTcpWrite(pProxyDevUsbIp->hSocket, &ReqUnlink, sizeof(ReqUnlink));
1559}
1560
1561static DECLCALLBACK(int) usbProxyUsbIpWakeup(PUSBPROXYDEV pProxyDev)
1562{
1563 LogFlowFunc(("pProxyDev=%s\n", pProxyDev->pUsbIns->pszName));
1564
1565 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1566 return usbProxyReaperKick(pProxyDevUsbIp, USBIP_REAPER_WAKEUP_REASON_EXTERNAL);
1567}
1568
1569/**
1570 * The USB/IP USB Proxy Backend operations.
1571 */
1572extern const USBPROXYBACK g_USBProxyDeviceUsbIp =
1573{
1574 /* pszName */
1575 "usbip",
1576 /* cbBackend */
1577 sizeof(USBPROXYDEVUSBIP),
1578 usbProxyUsbIpOpen,
1579 NULL,
1580 usbProxyUsbIpClose,
1581 usbProxyUsbIpReset,
1582 usbProxyUsbIpSetConfig,
1583 usbProxyUsbIpClaimInterface,
1584 usbProxyUsbIpReleaseInterface,
1585 usbProxyUsbIpSetInterface,
1586 usbProxyUsbIpClearHaltedEp,
1587 usbProxyUsbIpUrbQueue,
1588 usbProxyUsbIpUrbCancel,
1589 usbProxyUsbIpUrbReap,
1590 usbProxyUsbIpWakeup,
1591 0
1592};
1593
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