VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/UsbMsd.cpp@ 57034

Last change on this file since 57034 was 56992, checked in by vboxsync, 9 years ago

Devices: Log & Assertion formatting fixes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 81.2 KB
Line 
1/* $Id: UsbMsd.cpp 56992 2015-07-18 23:01:44Z vboxsync $ */
2/** @file
3 * UsbMSD - USB Mass Storage Device Emulation.
4 */
5
6/*
7 * Copyright (C) 2007-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* Header Files *
20*******************************************************************************/
21#define LOG_GROUP LOG_GROUP_USB_MSD
22#include <VBox/vmm/pdmusb.h>
23#include <VBox/log.h>
24#include <VBox/err.h>
25#include <VBox/scsi.h>
26#include <iprt/assert.h>
27#include <iprt/critsect.h>
28#include <iprt/mem.h>
29#include <iprt/semaphore.h>
30#include <iprt/string.h>
31#include <iprt/uuid.h>
32#include "VBoxDD.h"
33
34
35/*******************************************************************************
36* Defined Constants And Macros *
37*******************************************************************************/
38/** @name USB MSD string IDs
39 * @{ */
40#define USBMSD_STR_ID_MANUFACTURER 1
41#define USBMSD_STR_ID_PRODUCT_HD 2
42#define USBMSD_STR_ID_PRODUCT_CDROM 3
43/** @} */
44
45/** @name USB MSD vendor and product IDs
46 * @{ */
47#define VBOX_USB_VENDOR 0x80EE
48#define USBMSD_PID_HD 0x0030
49#define USBMSD_PID_CD 0x0031
50/** @} */
51
52/** Saved state version. */
53#define USB_MSD_SAVED_STATE_VERSION 1
54
55/*******************************************************************************
56* Structures and Typedefs *
57*******************************************************************************/
58
59/**
60 * USB MSD Command Block Wrapper or CBW. The command block
61 * itself (CBWCB) contains protocol-specific data (here SCSI).
62 */
63#pragma pack(1)
64typedef struct USBCBW
65{
66 uint32_t dCBWSignature;
67#define USBCBW_SIGNATURE UINT32_C(0x43425355)
68 uint32_t dCBWTag;
69 uint32_t dCBWDataTransferLength;
70 uint8_t bmCBWFlags;
71#define USBCBW_DIR_MASK RT_BIT(7)
72#define USBCBW_DIR_OUT 0
73#define USBCBW_DIR_IN RT_BIT(7)
74 uint8_t bCBWLun;
75 uint8_t bCBWCBLength;
76 uint8_t CBWCB[16];
77} USBCBW;
78#pragma pack()
79AssertCompileSize(USBCBW, 31);
80/** Pointer to a Command Block Wrapper. */
81typedef USBCBW *PUSBCBW;
82/** Pointer to a const Command Block Wrapper. */
83typedef const USBCBW *PCUSBCBW;
84
85/**
86 * USB MSD Command Status Wrapper or CSW.
87 */
88#pragma pack(1)
89typedef struct USBCSW
90{
91 uint32_t dCSWSignature;
92#define USBCSW_SIGNATURE UINT32_C(0x53425355)
93 uint32_t dCSWTag;
94 uint32_t dCSWDataResidue;
95#define USBCSW_STATUS_OK UINT8_C(0)
96#define USBCSW_STATUS_FAILED UINT8_C(1)
97#define USBCSW_STATUS_PHASE_ERROR UINT8_C(2)
98 uint8_t bCSWStatus;
99} USBCSW;
100#pragma pack()
101AssertCompileSize(USBCSW, 13);
102/** Pointer to a Command Status Wrapper. */
103typedef USBCSW *PUSBCSW;
104/** Pointer to a const Command Status Wrapper. */
105typedef const USBCSW *PCUSBCSW;
106
107
108/**
109 * The USB MSD request state.
110 */
111typedef enum USBMSDREQSTATE
112{
113 /** Invalid status. */
114 USBMSDREQSTATE_INVALID = 0,
115 /** Ready to receive a new SCSI command. */
116 USBMSDREQSTATE_READY,
117 /** Waiting for the host to supply data. */
118 USBMSDREQSTATE_DATA_FROM_HOST,
119 /** The SCSI request is being executed by the driver. */
120 USBMSDREQSTATE_EXECUTING,
121 /** Have (more) data for the host. */
122 USBMSDREQSTATE_DATA_TO_HOST,
123 /** Waiting to supply status information to the host. */
124 USBMSDREQSTATE_STATUS,
125 /** Destroy the request upon completion.
126 * This is set when the SCSI request doesn't complete before for the device or
127 * mass storage reset operation times out. USBMSD::pReq will be set to NULL
128 * and the only reference to this request will be with DrvSCSI. */
129 USBMSDREQSTATE_DESTROY_ON_COMPLETION,
130 /** The end of the valid states. */
131 USBMSDREQSTATE_END,
132 /** 32bit blow up hack. */
133 USBMSDREQSTATE_32BIT_HACK = 0x7fffffff
134} USBMSDREQSTATE;
135
136
137/**
138 * A pending USB MSD request.
139 */
140typedef struct USBMSDREQ
141{
142 /** The state of the request. */
143 USBMSDREQSTATE enmState;
144 /** The size of the data buffer. */
145 uint32_t cbBuf;
146 /** Pointer to the data buffer. */
147 uint8_t *pbBuf;
148 /** Current buffer offset. */
149 uint32_t offBuf;
150 /** The current Cbw when we're in the pending state. */
151 USBCBW Cbw;
152 /** The current SCSI request. */
153 PDMSCSIREQUEST ScsiReq;
154 /** The scatter-gather segment used by ScsiReq for describing pbBuf. */
155 RTSGSEG ScsiReqSeg;
156 /** The sense buffer for the current SCSI request. */
157 uint8_t ScsiReqSense[64];
158 /** The status of a completed SCSI request. */
159 int iScsiReqStatus;
160 /** Pointer to the USB device instance owning it. */
161 PPDMUSBINS pUsbIns;
162} USBMSDREQ;
163/** Pointer to a USB MSD request. */
164typedef USBMSDREQ *PUSBMSDREQ;
165
166
167/**
168 * Endpoint status data.
169 */
170typedef struct USBMSDEP
171{
172 bool fHalted;
173} USBMSDEP;
174/** Pointer to the endpoint status. */
175typedef USBMSDEP *PUSBMSDEP;
176
177
178/**
179 * A URB queue.
180 */
181typedef struct USBMSDURBQUEUE
182{
183 /** The head pointer. */
184 PVUSBURB pHead;
185 /** Where to insert the next entry. */
186 PVUSBURB *ppTail;
187} USBMSDURBQUEUE;
188/** Pointer to a URB queue. */
189typedef USBMSDURBQUEUE *PUSBMSDURBQUEUE;
190/** Pointer to a const URB queue. */
191typedef USBMSDURBQUEUE const *PCUSBMSDURBQUEUE;
192
193
194/**
195 * The USB MSD instance data.
196 */
197typedef struct USBMSD
198{
199 /** Pointer back to the PDM USB Device instance structure. */
200 PPDMUSBINS pUsbIns;
201 /** Critical section protecting the device state. */
202 RTCRITSECT CritSect;
203
204 /** The current configuration.
205 * (0 - default, 1 - the only, i.e configured.) */
206 uint8_t bConfigurationValue;
207#if 0
208 /** The state of the MSD (state machine).*/
209 USBMSDSTATE enmState;
210#endif
211 /** Endpoint 0 is the default control pipe, 1 is the host->dev bulk pipe and 2
212 * is the dev->host one. */
213 USBMSDEP aEps[3];
214 /** The current request. */
215 PUSBMSDREQ pReq;
216
217 /** Pending to-host queue.
218 * The URBs waiting here are pending the completion of the current request and
219 * data or status to become available.
220 */
221 USBMSDURBQUEUE ToHostQueue;
222
223 /** Done queue
224 * The URBs stashed here are waiting to be reaped. */
225 USBMSDURBQUEUE DoneQueue;
226 /** Signalled when adding an URB to the done queue and fHaveDoneQueueWaiter
227 * is set. */
228 RTSEMEVENT hEvtDoneQueue;
229 /** Someone is waiting on the done queue. */
230 bool fHaveDoneQueueWaiter;
231
232 /** Whether to signal the reset semaphore when the current request completes. */
233 bool fSignalResetSem;
234 /** Semaphore usbMsdUsbReset waits on when a request is executing at reset
235 * time. Only signalled when fSignalResetSem is set. */
236 RTSEMEVENTMULTI hEvtReset;
237 /** The reset URB.
238 * This is waiting for SCSI request completion before finishing the reset. */
239 PVUSBURB pResetUrb;
240 /** Indicates that PDMUsbHlpAsyncNotificationCompleted should be called when
241 * the MSD is entering the idle state. */
242 volatile bool fSignalIdle;
243
244 /** Indicates that this device is a CD-ROM. */
245 bool fIsCdrom;
246
247 /**
248 * LUN\#0 data.
249 */
250 struct
251 {
252 /** The base interface for LUN\#0. */
253 PDMIBASE IBase;
254 /** The SCSI port interface for LUN\#0 */
255 PDMISCSIPORT IScsiPort;
256
257 /** The base interface for the SCSI driver connected to LUN\#0. */
258 PPDMIBASE pIBase;
259 /** The SCSI connector interface for the SCSI driver connected to LUN\#0. */
260 PPDMISCSICONNECTOR pIScsiConnector;
261 } Lun0;
262
263} USBMSD;
264/** Pointer to the USB MSD instance data. */
265typedef USBMSD *PUSBMSD;
266
267
268/*******************************************************************************
269* Global Variables *
270*******************************************************************************/
271static const PDMUSBDESCCACHESTRING g_aUsbMsdStrings_en_US[] =
272{
273 { USBMSD_STR_ID_MANUFACTURER, "VirtualBox" },
274 { USBMSD_STR_ID_PRODUCT_HD, "USB Harddisk" },
275 { USBMSD_STR_ID_PRODUCT_CDROM, "USB CD-ROM" }
276};
277
278static const PDMUSBDESCCACHELANG g_aUsbMsdLanguages[] =
279{
280 { 0x0409, RT_ELEMENTS(g_aUsbMsdStrings_en_US), g_aUsbMsdStrings_en_US }
281};
282
283static const VUSBDESCENDPOINTEX g_aUsbMsdEndpointDescsFS[2] =
284{
285 {
286 {
287 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
288 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
289 /* .bEndpointAddress = */ 0x81 /* ep=1, in */,
290 /* .bmAttributes = */ 2 /* bulk */,
291 /* .wMaxPacketSize = */ 64 /* maximum possible */,
292 /* .bInterval = */ 0 /* not applicable for bulk EP */
293 },
294 /* .pvMore = */ NULL,
295 /* .pvClass = */ NULL,
296 /* .cbClass = */ 0,
297 /* .pvSsepc = */ NULL,
298 /* .cbSsepc = */ 0
299 },
300 {
301 {
302 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
303 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
304 /* .bEndpointAddress = */ 0x02 /* ep=2, out */,
305 /* .bmAttributes = */ 2 /* bulk */,
306 /* .wMaxPacketSize = */ 64 /* maximum possible */,
307 /* .bInterval = */ 0 /* not applicable for bulk EP */
308 },
309 /* .pvMore = */ NULL,
310 /* .pvClass = */ NULL,
311 /* .cbClass = */ 0,
312 /* .pvSsepc = */ NULL,
313 /* .cbSsepc = */ 0
314 }
315};
316
317static const VUSBDESCENDPOINTEX g_aUsbMsdEndpointDescsHS[2] =
318{
319 {
320 {
321 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
322 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
323 /* .bEndpointAddress = */ 0x81 /* ep=1, in */,
324 /* .bmAttributes = */ 2 /* bulk */,
325 /* .wMaxPacketSize = */ 512 /* HS bulk packet size */,
326 /* .bInterval = */ 0 /* no NAKs */
327 },
328 /* .pvMore = */ NULL,
329 /* .pvClass = */ NULL,
330 /* .cbClass = */ 0,
331 /* .pvSsepc = */ NULL,
332 /* .cbSsepc = */ 0
333 },
334 {
335 {
336 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
337 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
338 /* .bEndpointAddress = */ 0x02 /* ep=2, out */,
339 /* .bmAttributes = */ 2 /* bulk */,
340 /* .wMaxPacketSize = */ 512 /* HS bulk packet size */,
341 /* .bInterval = */ 0 /* no NAKs */
342 },
343 /* .pvMore = */ NULL,
344 /* .pvClass = */ NULL,
345 /* .cbClass = */ 0,
346 /* .pvSsepc = */ NULL,
347 /* .cbSsepc = */ 0
348 }
349};
350
351static const VUSBDESCSSEPCOMPANION g_aUsbMsdEpCompanionSS =
352{
353 /* .bLength = */ sizeof(VUSBDESCSSEPCOMPANION),
354 /* .bDescriptorType = */ VUSB_DT_SS_ENDPOINT_COMPANION,
355 /* .bMaxBurst = */ 15 /* we can burst all the way */,
356 /* .bmAttributes = */ 0 /* no streams */,
357 /* .wBytesPerInterval = */ 0 /* not a periodic endpoint */
358};
359
360static const VUSBDESCENDPOINTEX g_aUsbMsdEndpointDescsSS[2] =
361{
362 {
363 {
364 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
365 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
366 /* .bEndpointAddress = */ 0x81 /* ep=1, in */,
367 /* .bmAttributes = */ 2 /* bulk */,
368 /* .wMaxPacketSize = */ 1024 /* SS bulk packet size */,
369 /* .bInterval = */ 0 /* no NAKs */
370 },
371 /* .pvMore = */ NULL,
372 /* .pvClass = */ NULL,
373 /* .cbClass = */ 0,
374 /* .pvSsepc = */ &g_aUsbMsdEpCompanionSS,
375 /* .cbSsepc = */ sizeof(g_aUsbMsdEpCompanionSS)
376 },
377 {
378 {
379 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
380 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
381 /* .bEndpointAddress = */ 0x02 /* ep=2, out */,
382 /* .bmAttributes = */ 2 /* bulk */,
383 /* .wMaxPacketSize = */ 1024 /* SS bulk packet size */,
384 /* .bInterval = */ 0 /* no NAKs */
385 },
386 /* .pvMore = */ NULL,
387 /* .pvClass = */ NULL,
388 /* .cbClass = */ 0,
389 /* .pvSsepc = */ &g_aUsbMsdEpCompanionSS,
390 /* .cbSsepc = */ sizeof(g_aUsbMsdEpCompanionSS)
391 }
392};
393
394static const VUSBDESCINTERFACEEX g_UsbMsdInterfaceDescFS =
395{
396 {
397 /* .bLength = */ sizeof(VUSBDESCINTERFACE),
398 /* .bDescriptorType = */ VUSB_DT_INTERFACE,
399 /* .bInterfaceNumber = */ 0,
400 /* .bAlternateSetting = */ 0,
401 /* .bNumEndpoints = */ 2,
402 /* .bInterfaceClass = */ 8 /* Mass Storage */,
403 /* .bInterfaceSubClass = */ 6 /* SCSI transparent command set */,
404 /* .bInterfaceProtocol = */ 0x50 /* Bulk-Only Transport */,
405 /* .iInterface = */ 0
406 },
407 /* .pvMore = */ NULL,
408 /* .pvClass = */ NULL,
409 /* .cbClass = */ 0,
410 &g_aUsbMsdEndpointDescsFS[0],
411 /* .pIAD = */ NULL,
412 /* .cbIAD = */ 0
413};
414
415static const VUSBDESCINTERFACEEX g_UsbMsdInterfaceDescHS =
416{
417 {
418 /* .bLength = */ sizeof(VUSBDESCINTERFACE),
419 /* .bDescriptorType = */ VUSB_DT_INTERFACE,
420 /* .bInterfaceNumber = */ 0,
421 /* .bAlternateSetting = */ 0,
422 /* .bNumEndpoints = */ 2,
423 /* .bInterfaceClass = */ 8 /* Mass Storage */,
424 /* .bInterfaceSubClass = */ 6 /* SCSI transparent command set */,
425 /* .bInterfaceProtocol = */ 0x50 /* Bulk-Only Transport */,
426 /* .iInterface = */ 0
427 },
428 /* .pvMore = */ NULL,
429 /* .pvClass = */ NULL,
430 /* .cbClass = */ 0,
431 &g_aUsbMsdEndpointDescsHS[0],
432 /* .pIAD = */ NULL,
433 /* .cbIAD = */ 0
434};
435
436static const VUSBDESCINTERFACEEX g_UsbMsdInterfaceDescSS =
437{
438 {
439 /* .bLength = */ sizeof(VUSBDESCINTERFACE),
440 /* .bDescriptorType = */ VUSB_DT_INTERFACE,
441 /* .bInterfaceNumber = */ 0,
442 /* .bAlternateSetting = */ 0,
443 /* .bNumEndpoints = */ 2,
444 /* .bInterfaceClass = */ 8 /* Mass Storage */,
445 /* .bInterfaceSubClass = */ 6 /* SCSI transparent command set */,
446 /* .bInterfaceProtocol = */ 0x50 /* Bulk-Only Transport */,
447 /* .iInterface = */ 0
448 },
449 /* .pvMore = */ NULL,
450 /* .pvClass = */ NULL,
451 /* .cbClass = */ 0,
452 &g_aUsbMsdEndpointDescsSS[0],
453 /* .pIAD = */ NULL,
454 /* .cbIAD = */ 0
455};
456
457static const VUSBINTERFACE g_aUsbMsdInterfacesFS[] =
458{
459 { &g_UsbMsdInterfaceDescFS, /* .cSettings = */ 1 },
460};
461
462static const VUSBINTERFACE g_aUsbMsdInterfacesHS[] =
463{
464 { &g_UsbMsdInterfaceDescHS, /* .cSettings = */ 1 },
465};
466
467static const VUSBINTERFACE g_aUsbMsdInterfacesSS[] =
468{
469 { &g_UsbMsdInterfaceDescSS, /* .cSettings = */ 1 },
470};
471
472static const VUSBDESCCONFIGEX g_UsbMsdConfigDescFS =
473{
474 {
475 /* .bLength = */ sizeof(VUSBDESCCONFIG),
476 /* .bDescriptorType = */ VUSB_DT_CONFIG,
477 /* .wTotalLength = */ 0 /* recalculated on read */,
478 /* .bNumInterfaces = */ RT_ELEMENTS(g_aUsbMsdInterfacesFS),
479 /* .bConfigurationValue =*/ 1,
480 /* .iConfiguration = */ 0,
481 /* .bmAttributes = */ RT_BIT(7),
482 /* .MaxPower = */ 50 /* 100mA */
483 },
484 NULL, /* pvMore */
485 &g_aUsbMsdInterfacesFS[0],
486 NULL /* pvOriginal */
487};
488
489static const VUSBDESCCONFIGEX g_UsbMsdConfigDescHS =
490{
491 {
492 /* .bLength = */ sizeof(VUSBDESCCONFIG),
493 /* .bDescriptorType = */ VUSB_DT_CONFIG,
494 /* .wTotalLength = */ 0 /* recalculated on read */,
495 /* .bNumInterfaces = */ RT_ELEMENTS(g_aUsbMsdInterfacesHS),
496 /* .bConfigurationValue =*/ 1,
497 /* .iConfiguration = */ 0,
498 /* .bmAttributes = */ RT_BIT(7),
499 /* .MaxPower = */ 50 /* 100mA */
500 },
501 NULL, /* pvMore */
502 &g_aUsbMsdInterfacesHS[0],
503 NULL /* pvOriginal */
504};
505
506static const VUSBDESCCONFIGEX g_UsbMsdConfigDescSS =
507{
508 {
509 /* .bLength = */ sizeof(VUSBDESCCONFIG),
510 /* .bDescriptorType = */ VUSB_DT_CONFIG,
511 /* .wTotalLength = */ 0 /* recalculated on read */,
512 /* .bNumInterfaces = */ RT_ELEMENTS(g_aUsbMsdInterfacesSS),
513 /* .bConfigurationValue =*/ 1,
514 /* .iConfiguration = */ 0,
515 /* .bmAttributes = */ RT_BIT(7),
516 /* .MaxPower = */ 50 /* 100mA */
517 },
518 NULL, /* pvMore */
519 &g_aUsbMsdInterfacesSS[0],
520 NULL /* pvOriginal */
521};
522
523static const VUSBDESCDEVICE g_UsbMsdDeviceDesc20 =
524{
525 /* .bLength = */ sizeof(g_UsbMsdDeviceDesc20),
526 /* .bDescriptorType = */ VUSB_DT_DEVICE,
527 /* .bcdUsb = */ 0x200, /* USB 2.0 */
528 /* .bDeviceClass = */ 0 /* Class specified in the interface desc. */,
529 /* .bDeviceSubClass = */ 0 /* Subclass specified in the interface desc. */,
530 /* .bDeviceProtocol = */ 0 /* Protocol specified in the interface desc. */,
531 /* .bMaxPacketSize0 = */ 64,
532 /* .idVendor = */ VBOX_USB_VENDOR,
533 /* .idProduct = */ USBMSD_PID_HD,
534 /* .bcdDevice = */ 0x0100, /* 1.0 */
535 /* .iManufacturer = */ USBMSD_STR_ID_MANUFACTURER,
536 /* .iProduct = */ USBMSD_STR_ID_PRODUCT_HD,
537 /* .iSerialNumber = */ 0,
538 /* .bNumConfigurations = */ 1
539};
540
541static const VUSBDESCDEVICE g_UsbCdDeviceDesc20 =
542{
543 /* .bLength = */ sizeof(g_UsbCdDeviceDesc20),
544 /* .bDescriptorType = */ VUSB_DT_DEVICE,
545 /* .bcdUsb = */ 0x200, /* USB 2.0 */
546 /* .bDeviceClass = */ 0 /* Class specified in the interface desc. */,
547 /* .bDeviceSubClass = */ 0 /* Subclass specified in the interface desc. */,
548 /* .bDeviceProtocol = */ 0 /* Protocol specified in the interface desc. */,
549 /* .bMaxPacketSize0 = */ 64,
550 /* .idVendor = */ VBOX_USB_VENDOR,
551 /* .idProduct = */ USBMSD_PID_CD,
552 /* .bcdDevice = */ 0x0100, /* 1.0 */
553 /* .iManufacturer = */ USBMSD_STR_ID_MANUFACTURER,
554 /* .iProduct = */ USBMSD_STR_ID_PRODUCT_CDROM,
555 /* .iSerialNumber = */ 0,
556 /* .bNumConfigurations = */ 1
557};
558
559static const VUSBDESCDEVICE g_UsbMsdDeviceDesc30 =
560{
561 /* .bLength = */ sizeof(g_UsbMsdDeviceDesc30),
562 /* .bDescriptorType = */ VUSB_DT_DEVICE,
563 /* .bcdUsb = */ 0x300, /* USB 2.0 */
564 /* .bDeviceClass = */ 0 /* Class specified in the interface desc. */,
565 /* .bDeviceSubClass = */ 0 /* Subclass specified in the interface desc. */,
566 /* .bDeviceProtocol = */ 0 /* Protocol specified in the interface desc. */,
567 /* .bMaxPacketSize0 = */ 9 /* 512, the only option for USB3. */,
568 /* .idVendor = */ VBOX_USB_VENDOR,
569 /* .idProduct = */ USBMSD_PID_HD,
570 /* .bcdDevice = */ 0x0110, /* 1.10 */
571 /* .iManufacturer = */ USBMSD_STR_ID_MANUFACTURER,
572 /* .iProduct = */ USBMSD_STR_ID_PRODUCT_HD,
573 /* .iSerialNumber = */ 0,
574 /* .bNumConfigurations = */ 1
575};
576
577static const VUSBDESCDEVICE g_UsbCdDeviceDesc30 =
578{
579 /* .bLength = */ sizeof(g_UsbCdDeviceDesc30),
580 /* .bDescriptorType = */ VUSB_DT_DEVICE,
581 /* .bcdUsb = */ 0x300, /* USB 2.0 */
582 /* .bDeviceClass = */ 0 /* Class specified in the interface desc. */,
583 /* .bDeviceSubClass = */ 0 /* Subclass specified in the interface desc. */,
584 /* .bDeviceProtocol = */ 0 /* Protocol specified in the interface desc. */,
585 /* .bMaxPacketSize0 = */ 9 /* 512, the only option for USB3. */,
586 /* .idVendor = */ VBOX_USB_VENDOR,
587 /* .idProduct = */ USBMSD_PID_CD,
588 /* .bcdDevice = */ 0x0110, /* 1.10 */
589 /* .iManufacturer = */ USBMSD_STR_ID_MANUFACTURER,
590 /* .iProduct = */ USBMSD_STR_ID_PRODUCT_CDROM,
591 /* .iSerialNumber = */ 0,
592 /* .bNumConfigurations = */ 1
593};
594
595static const VUSBDEVICEQUALIFIER g_UsbMsdDeviceQualifier =
596{
597 /* .bLength = */ sizeof(g_UsbMsdDeviceQualifier),
598 /* .bDescriptorType = */ VUSB_DT_DEVICE_QUALIFIER,
599 /* .bcdUsb = */ 0x200, /* USB 2.0 */
600 /* .bDeviceClass = */ 0 /* Class specified in the interface desc. */,
601 /* .bDeviceSubClass = */ 0 /* Subclass specified in the interface desc. */,
602 /* .bDeviceProtocol = */ 0 /* Protocol specified in the interface desc. */,
603 /* .bMaxPacketSize0 = */ 64,
604 /* .bNumConfigurations = */ 1,
605 /* .bReserved = */ 0
606};
607
608static const struct {
609 VUSBDESCBOS bos;
610 VUSBDESCSSDEVCAP sscap;
611} g_UsbMsdBOS =
612{
613 {
614 /* .bLength = */ sizeof(g_UsbMsdBOS.bos),
615 /* .bDescriptorType = */ VUSB_DT_BOS,
616 /* .wTotalLength = */ sizeof(g_UsbMsdBOS),
617 /* .bNumDeviceCaps = */ 1
618 },
619 {
620 /* .bLength = */ sizeof(VUSBDESCSSDEVCAP),
621 /* .bDescriptorType = */ VUSB_DT_DEVICE_CAPABILITY,
622 /* .bDevCapabilityType = */ VUSB_DCT_SUPERSPEED_USB,
623 /* .bmAttributes = */ 0 /* No LTM. */,
624 /* .wSpeedsSupported = */ 0xe /* Any speed is good. */,
625 /* .bFunctionalitySupport = */ 2 /* Want HS at least. */,
626 /* .bU1DevExitLat = */ 0, /* We are blazingly fast. */
627 /* .wU2DevExitLat = */ 0
628 }
629};
630
631static const PDMUSBDESCCACHE g_UsbMsdDescCacheFS =
632{
633 /* .pDevice = */ &g_UsbMsdDeviceDesc20,
634 /* .paConfigs = */ &g_UsbMsdConfigDescFS,
635 /* .paLanguages = */ g_aUsbMsdLanguages,
636 /* .cLanguages = */ RT_ELEMENTS(g_aUsbMsdLanguages),
637 /* .fUseCachedDescriptors = */ true,
638 /* .fUseCachedStringsDescriptors = */ true
639};
640
641static const PDMUSBDESCCACHE g_UsbCdDescCacheFS =
642{
643 /* .pDevice = */ &g_UsbCdDeviceDesc20,
644 /* .paConfigs = */ &g_UsbMsdConfigDescFS,
645 /* .paLanguages = */ g_aUsbMsdLanguages,
646 /* .cLanguages = */ RT_ELEMENTS(g_aUsbMsdLanguages),
647 /* .fUseCachedDescriptors = */ true,
648 /* .fUseCachedStringsDescriptors = */ true
649};
650
651static const PDMUSBDESCCACHE g_UsbMsdDescCacheHS =
652{
653 /* .pDevice = */ &g_UsbMsdDeviceDesc20,
654 /* .paConfigs = */ &g_UsbMsdConfigDescHS,
655 /* .paLanguages = */ g_aUsbMsdLanguages,
656 /* .cLanguages = */ RT_ELEMENTS(g_aUsbMsdLanguages),
657 /* .fUseCachedDescriptors = */ true,
658 /* .fUseCachedStringsDescriptors = */ true
659};
660
661static const PDMUSBDESCCACHE g_UsbCdDescCacheHS =
662{
663 /* .pDevice = */ &g_UsbCdDeviceDesc20,
664 /* .paConfigs = */ &g_UsbMsdConfigDescHS,
665 /* .paLanguages = */ g_aUsbMsdLanguages,
666 /* .cLanguages = */ RT_ELEMENTS(g_aUsbMsdLanguages),
667 /* .fUseCachedDescriptors = */ true,
668 /* .fUseCachedStringsDescriptors = */ true
669};
670
671static const PDMUSBDESCCACHE g_UsbMsdDescCacheSS =
672{
673 /* .pDevice = */ &g_UsbMsdDeviceDesc30,
674 /* .paConfigs = */ &g_UsbMsdConfigDescSS,
675 /* .paLanguages = */ g_aUsbMsdLanguages,
676 /* .cLanguages = */ RT_ELEMENTS(g_aUsbMsdLanguages),
677 /* .fUseCachedDescriptors = */ true,
678 /* .fUseCachedStringsDescriptors = */ true
679};
680
681static const PDMUSBDESCCACHE g_UsbCdDescCacheSS =
682{
683 /* .pDevice = */ &g_UsbCdDeviceDesc30,
684 /* .paConfigs = */ &g_UsbMsdConfigDescSS,
685 /* .paLanguages = */ g_aUsbMsdLanguages,
686 /* .cLanguages = */ RT_ELEMENTS(g_aUsbMsdLanguages),
687 /* .fUseCachedDescriptors = */ true,
688 /* .fUseCachedStringsDescriptors = */ true
689};
690
691/*******************************************************************************
692* Internal Functions *
693*******************************************************************************/
694static int usbMsdHandleBulkDevToHost(PUSBMSD pThis, PUSBMSDEP pEp, PVUSBURB pUrb);
695
696
697/**
698 * Initializes an URB queue.
699 *
700 * @param pQueue The URB queue.
701 */
702static void usbMsdQueueInit(PUSBMSDURBQUEUE pQueue)
703{
704 pQueue->pHead = NULL;
705 pQueue->ppTail = &pQueue->pHead;
706}
707
708
709
710/**
711 * Inserts an URB at the end of the queue.
712 *
713 * @param pQueue The URB queue.
714 * @param pUrb The URB to insert.
715 */
716DECLINLINE(void) usbMsdQueueAddTail(PUSBMSDURBQUEUE pQueue, PVUSBURB pUrb)
717{
718 pUrb->Dev.pNext = NULL;
719 *pQueue->ppTail = pUrb;
720 pQueue->ppTail = &pUrb->Dev.pNext;
721}
722
723
724/**
725 * Unlinks the head of the queue and returns it.
726 *
727 * @returns The head entry.
728 * @param pQueue The URB queue.
729 */
730DECLINLINE(PVUSBURB) usbMsdQueueRemoveHead(PUSBMSDURBQUEUE pQueue)
731{
732 PVUSBURB pUrb = pQueue->pHead;
733 if (pUrb)
734 {
735 PVUSBURB pNext = pUrb->Dev.pNext;
736 pQueue->pHead = pNext;
737 if (!pNext)
738 pQueue->ppTail = &pQueue->pHead;
739 else
740 pUrb->Dev.pNext = NULL;
741 }
742 return pUrb;
743}
744
745
746/**
747 * Removes an URB from anywhere in the queue.
748 *
749 * @returns true if found, false if not.
750 * @param pQueue The URB queue.
751 * @param pUrb The URB to remove.
752 */
753DECLINLINE(bool) usbMsdQueueRemove(PUSBMSDURBQUEUE pQueue, PVUSBURB pUrb)
754{
755 PVUSBURB pCur = pQueue->pHead;
756 if (pCur == pUrb)
757 pQueue->pHead = pUrb->Dev.pNext;
758 else
759 {
760 while (pCur)
761 {
762 if (pCur->Dev.pNext == pUrb)
763 {
764 pCur->Dev.pNext = pUrb->Dev.pNext;
765 break;
766 }
767 pCur = pCur->Dev.pNext;
768 }
769 if (!pCur)
770 return false;
771 }
772 if (!pUrb->Dev.pNext)
773 pQueue->ppTail = &pQueue->pHead;
774 return true;
775}
776
777
778/**
779 * Checks if the queue is empty or not.
780 *
781 * @returns true if it is, false if it isn't.
782 * @param pQueue The URB queue.
783 */
784DECLINLINE(bool) usbMsdQueueIsEmpty(PCUSBMSDURBQUEUE pQueue)
785{
786 return pQueue->pHead == NULL;
787}
788
789
790/**
791 * Links an URB into the done queue.
792 *
793 * @param pThis The MSD instance.
794 * @param pUrb The URB.
795 */
796static void usbMsdLinkDone(PUSBMSD pThis, PVUSBURB pUrb)
797{
798 usbMsdQueueAddTail(&pThis->DoneQueue, pUrb);
799
800 if (pThis->fHaveDoneQueueWaiter)
801 {
802 int rc = RTSemEventSignal(pThis->hEvtDoneQueue);
803 AssertRC(rc);
804 }
805}
806
807
808
809
810/**
811 * Allocates a new request and does basic init.
812 *
813 * @returns Pointer to the new request. NULL if we're out of memory.
814 * @param pUsbIns The instance allocating it.
815 */
816static PUSBMSDREQ usbMsdReqAlloc(PPDMUSBINS pUsbIns)
817{
818 PUSBMSDREQ pReq = (PUSBMSDREQ)PDMUsbHlpMMHeapAllocZ(pUsbIns, sizeof(*pReq));
819 if (pReq)
820 {
821 pReq->enmState = USBMSDREQSTATE_READY;
822 pReq->iScsiReqStatus = -1;
823 pReq->pUsbIns = pUsbIns;
824 }
825 else
826 LogRel(("usbMsdReqAlloc: Out of memory\n"));
827 return pReq;
828}
829
830
831/**
832 * Frees a request.
833 *
834 * @param pReq The request.
835 */
836static void usbMsdReqFree(PUSBMSDREQ pReq)
837{
838 /*
839 * Check the input.
840 */
841 AssertReturnVoid( pReq->enmState > USBMSDREQSTATE_INVALID
842 && pReq->enmState != USBMSDREQSTATE_EXECUTING
843 && pReq->enmState < USBMSDREQSTATE_END);
844 PPDMUSBINS pUsbIns = pReq->pUsbIns;
845 AssertPtrReturnVoid(pUsbIns);
846 AssertReturnVoid(PDM_VERSION_ARE_COMPATIBLE(pUsbIns->u32Version, PDM_USBINS_VERSION));
847
848 /*
849 * Invalidate it and free the associated resources.
850 */
851 pReq->enmState = USBMSDREQSTATE_INVALID;
852 pReq->cbBuf = 0;
853 pReq->offBuf = 0;
854 pReq->ScsiReq.pbCDB = NULL;
855 pReq->ScsiReq.paScatterGatherHead = NULL;
856 pReq->ScsiReq.pbSenseBuffer = NULL;
857 pReq->ScsiReq.pvUser = NULL;
858 pReq->ScsiReqSeg.cbSeg = 0;
859 pReq->ScsiReqSeg.pvSeg = NULL;
860
861 if (pReq->pbBuf)
862 {
863 PDMUsbHlpMMHeapFree(pUsbIns, pReq->pbBuf);
864 pReq->pbBuf = NULL;
865 }
866
867 PDMUsbHlpMMHeapFree(pUsbIns, pReq);
868}
869
870
871/**
872 * Prepares a request for execution or data buffering.
873 *
874 * @param pReq The request.
875 * @param pCbw The SCSI command block wrapper.
876 */
877static void usbMsdReqPrepare(PUSBMSDREQ pReq, PCUSBCBW pCbw)
878{
879 /* Copy the CBW */
880 size_t cbCopy = RT_OFFSETOF(USBCBW, CBWCB[pCbw->bCBWCBLength]);
881 memcpy(&pReq->Cbw, pCbw, cbCopy);
882 memset((uint8_t *)&pReq->Cbw + cbCopy, 0, sizeof(pReq->Cbw) - cbCopy);
883
884 /* Setup the SCSI request. */
885 pReq->ScsiReq.uLogicalUnit = pReq->Cbw.bCBWLun;
886 pReq->ScsiReq.uDataDirection = (pReq->Cbw.bmCBWFlags & USBCBW_DIR_MASK) == USBCBW_DIR_OUT
887 ? PDMSCSIREQUESTTXDIR_TO_DEVICE
888 : PDMSCSIREQUESTTXDIR_FROM_DEVICE;
889 pReq->ScsiReq.cbCDB = pReq->Cbw.bCBWCBLength;
890
891 pReq->ScsiReq.pbCDB = &pReq->Cbw.CBWCB[0];
892 pReq->offBuf = 0;
893 pReq->ScsiReqSeg.pvSeg = pReq->pbBuf;
894 pReq->ScsiReqSeg.cbSeg = pReq->Cbw.dCBWDataTransferLength;
895 pReq->ScsiReq.cbScatterGather = pReq->Cbw.dCBWDataTransferLength;
896 pReq->ScsiReq.cScatterGatherEntries = 1;
897 pReq->ScsiReq.paScatterGatherHead = &pReq->ScsiReqSeg;
898 pReq->ScsiReq.cbSenseBuffer = sizeof(pReq->ScsiReqSense);
899 pReq->ScsiReq.pbSenseBuffer = &pReq->ScsiReqSense[0];
900 pReq->ScsiReq.pvUser = NULL;
901 RT_ZERO(pReq->ScsiReqSense);
902 pReq->iScsiReqStatus = -1;
903}
904
905
906/**
907 * Makes sure that there is sufficient buffer space available.
908 *
909 * @returns Success indicator (true/false)
910 * @param pReq
911 * @param cbBuf The required buffer space.
912 */
913static int usbMsdReqEnsureBuffer(PUSBMSDREQ pReq, uint32_t cbBuf)
914{
915 if (RT_LIKELY(pReq->cbBuf >= cbBuf))
916 RT_BZERO(pReq->pbBuf, cbBuf);
917 else
918 {
919 PDMUsbHlpMMHeapFree(pReq->pUsbIns, pReq->pbBuf);
920 pReq->cbBuf = 0;
921
922 cbBuf = RT_ALIGN_Z(cbBuf, 0x1000);
923 pReq->pbBuf = (uint8_t *)PDMUsbHlpMMHeapAllocZ(pReq->pUsbIns, cbBuf);
924 if (!pReq->pbBuf)
925 return false;
926
927 pReq->cbBuf = cbBuf;
928 }
929 return true;
930}
931
932
933/**
934 * Completes the URB with a stalled state, halting the pipe.
935 */
936static int usbMsdCompleteStall(PUSBMSD pThis, PUSBMSDEP pEp, PVUSBURB pUrb, const char *pszWhy)
937{
938 Log(("usbMsdCompleteStall/#%u: pUrb=%p:%s: %s\n", pThis->pUsbIns->iInstance, pUrb, pUrb->pszDesc, pszWhy));
939
940 pUrb->enmStatus = VUSBSTATUS_STALL;
941
942 /** @todo figure out if the stall is global or pipe-specific or both. */
943 if (pEp)
944 pEp->fHalted = true;
945 else
946 {
947 pThis->aEps[1].fHalted = true;
948 pThis->aEps[2].fHalted = true;
949 }
950
951 usbMsdLinkDone(pThis, pUrb);
952 return VINF_SUCCESS;
953}
954
955
956/**
957 * Completes the URB with a OK state.
958 */
959static int usbMsdCompleteOk(PUSBMSD pThis, PVUSBURB pUrb, size_t cbData)
960{
961 Log(("usbMsdCompleteOk/#%u: pUrb=%p:%s cbData=%#zx\n", pThis->pUsbIns->iInstance, pUrb, pUrb->pszDesc, cbData));
962
963 pUrb->enmStatus = VUSBSTATUS_OK;
964 pUrb->cbData = (uint32_t)cbData;
965
966 usbMsdLinkDone(pThis, pUrb);
967 return VINF_SUCCESS;
968}
969
970
971/**
972 * Reset worker for usbMsdUsbReset, usbMsdUsbSetConfiguration and
973 * usbMsdUrbHandleDefaultPipe.
974 *
975 * @returns VBox status code.
976 * @param pThis The MSD instance.
977 * @param pUrb Set when usbMsdUrbHandleDefaultPipe is the
978 * caller.
979 * @param fSetConfig Set when usbMsdUsbSetConfiguration is the
980 * caller.
981 */
982static int usbMsdResetWorker(PUSBMSD pThis, PVUSBURB pUrb, bool fSetConfig)
983{
984 /*
985 * Wait for the any command currently executing to complete before
986 * resetting. (We cannot cancel its execution.) How we do this depends
987 * on the reset method.
988 */
989 PUSBMSDREQ pReq = pThis->pReq;
990 if ( pReq
991 && pReq->enmState == USBMSDREQSTATE_EXECUTING)
992 {
993 /* Don't try to deal with the set config variant nor multiple build-only
994 mass storage resets. */
995 if (pThis->pResetUrb && (pUrb || fSetConfig))
996 {
997 Log(("usbMsdResetWorker: pResetUrb is already %p:%s - stalling\n", pThis->pResetUrb, pThis->pResetUrb->pszDesc));
998 return usbMsdCompleteStall(pThis, NULL, pUrb, "pResetUrb");
999 }
1000
1001 /* Bulk-Only Mass Storage Reset: Complete the reset on request completion. */
1002 if (pUrb)
1003 {
1004 pThis->pResetUrb = pUrb;
1005 Log(("usbMsdResetWorker: Setting pResetUrb to %p:%s\n", pThis->pResetUrb, pThis->pResetUrb->pszDesc));
1006 return VINF_SUCCESS;
1007 }
1008
1009 /* Device reset: Wait for up to 10 ms. If it doesn't work, ditch
1010 whoe the request structure. We'll allocate a new one when needed. */
1011 Log(("usbMsdResetWorker: Waiting for completion...\n"));
1012 Assert(!pThis->fSignalResetSem);
1013 pThis->fSignalResetSem = true;
1014 RTSemEventMultiReset(pThis->hEvtReset);
1015 RTCritSectLeave(&pThis->CritSect);
1016
1017 int rc = RTSemEventMultiWait(pThis->hEvtReset, 10 /*ms*/);
1018
1019 RTCritSectEnter(&pThis->CritSect);
1020 pThis->fSignalResetSem = false;
1021 if ( RT_FAILURE(rc)
1022 || pReq->enmState == USBMSDREQSTATE_EXECUTING)
1023 {
1024 Log(("usbMsdResetWorker: Didn't complete, ditching the current request (%p)!\n", pReq));
1025 Assert(pReq == pThis->pReq);
1026 pReq->enmState = USBMSDREQSTATE_DESTROY_ON_COMPLETION;
1027 pThis->pReq = NULL;
1028 pReq = NULL;
1029 }
1030 }
1031
1032 /*
1033 * Reset the request and device state.
1034 */
1035 if (pReq)
1036 {
1037 pReq->enmState = USBMSDREQSTATE_READY;
1038 pReq->iScsiReqStatus = -1;
1039 }
1040
1041 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aEps); i++)
1042 pThis->aEps[i].fHalted = false;
1043
1044 if (!pUrb && !fSetConfig) /* (only device reset) */
1045 pThis->bConfigurationValue = 0; /* default */
1046
1047 /*
1048 * Ditch all pending URBs.
1049 */
1050 PVUSBURB pCurUrb;
1051 while ((pCurUrb = usbMsdQueueRemoveHead(&pThis->ToHostQueue)) != NULL)
1052 {
1053 pCurUrb->enmStatus = VUSBSTATUS_CRC;
1054 usbMsdLinkDone(pThis, pCurUrb);
1055 }
1056
1057 pCurUrb = pThis->pResetUrb;
1058 if (pCurUrb)
1059 {
1060 pThis->pResetUrb = NULL;
1061 pCurUrb->enmStatus = VUSBSTATUS_CRC;
1062 usbMsdLinkDone(pThis, pCurUrb);
1063 }
1064
1065 if (pUrb)
1066 return usbMsdCompleteOk(pThis, pUrb, 0);
1067 return VINF_SUCCESS;
1068}
1069
1070
1071/**
1072 * @interface_method_impl{PDMISCSIPORT,pfnSCSIRequestCompleted}
1073 */
1074static DECLCALLBACK(int) usbMsdLun0ScsiRequestCompleted(PPDMISCSIPORT pInterface, PPDMSCSIREQUEST pSCSIRequest,
1075 int rcCompletion, bool fRedo, int rcReq)
1076{
1077 PUSBMSD pThis = RT_FROM_MEMBER(pInterface, USBMSD, Lun0.IScsiPort);
1078 PUSBMSDREQ pReq = RT_FROM_MEMBER(pSCSIRequest, USBMSDREQ, ScsiReq);
1079
1080 Log(("usbMsdLun0ScsiRequestCompleted: pReq=%p dCBWTag=%#x iScsiReqStatus=%u \n", pReq, pReq->Cbw.dCBWTag, rcCompletion));
1081 RTCritSectEnter(&pThis->CritSect);
1082
1083 if (pReq->enmState != USBMSDREQSTATE_DESTROY_ON_COMPLETION)
1084 {
1085 Assert(pReq->enmState == USBMSDREQSTATE_EXECUTING);
1086 Assert(pThis->pReq == pReq);
1087 pReq->iScsiReqStatus = rcCompletion;
1088
1089 /*
1090 * Advance the state machine. The state machine is not affected by
1091 * SCSI errors.
1092 */
1093 if ((pReq->Cbw.bmCBWFlags & USBCBW_DIR_MASK) == USBCBW_DIR_OUT)
1094 {
1095 pReq->enmState = USBMSDREQSTATE_STATUS;
1096 Log(("usbMsdLun0ScsiRequestCompleted: Entering STATUS\n"));
1097 }
1098 else
1099 {
1100 pReq->enmState = USBMSDREQSTATE_DATA_TO_HOST;
1101 Log(("usbMsdLun0ScsiRequestCompleted: Entering DATA_TO_HOST\n"));
1102 }
1103
1104 /*
1105 * Deal with pending to-host URBs.
1106 */
1107 for (;;)
1108 {
1109 PVUSBURB pUrb = usbMsdQueueRemoveHead(&pThis->ToHostQueue);
1110 if (!pUrb)
1111 break;
1112
1113 /* Process it the normal way. */
1114 usbMsdHandleBulkDevToHost(pThis, &pThis->aEps[1], pUrb);
1115 }
1116 }
1117 else
1118 {
1119 Log(("usbMsdLun0ScsiRequestCompleted: freeing %p\n", pReq));
1120 usbMsdReqFree(pReq);
1121 }
1122
1123 if (pThis->fSignalResetSem)
1124 RTSemEventMultiSignal(pThis->hEvtReset);
1125
1126 if (pThis->pResetUrb)
1127 {
1128 pThis->pResetUrb = NULL;
1129 usbMsdResetWorker(pThis, pThis->pResetUrb, false /*fSetConfig*/);
1130 }
1131
1132 RTCritSectLeave(&pThis->CritSect);
1133 return VINF_SUCCESS;
1134}
1135
1136
1137/**
1138 * @interface_method_impl{PDMISCSIPORT,pfnQueryDeviceLocation}
1139 */
1140static DECLCALLBACK(int) usbMsdLun0QueryDeviceLocation(PPDMISCSIPORT pInterface, const char **ppcszController,
1141 uint32_t *piInstance, uint32_t *piLUN)
1142{
1143 PUSBMSD pThis = RT_FROM_MEMBER(pInterface, USBMSD, Lun0.IScsiPort);
1144 PPDMUSBINS pUsbIns = pThis->pUsbIns;
1145
1146 AssertPtrReturn(ppcszController, VERR_INVALID_POINTER);
1147 AssertPtrReturn(piInstance, VERR_INVALID_POINTER);
1148 AssertPtrReturn(piLUN, VERR_INVALID_POINTER);
1149
1150 *ppcszController = pUsbIns->pReg->szName;
1151 *piInstance = pUsbIns->iInstance;
1152 *piLUN = 0;
1153
1154 return VINF_SUCCESS;
1155}
1156
1157
1158/**
1159 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1160 */
1161static DECLCALLBACK(void *) usbMsdLun0QueryInterface(PPDMIBASE pInterface, const char *pszIID)
1162{
1163 PUSBMSD pThis = RT_FROM_MEMBER(pInterface, USBMSD, Lun0.IBase);
1164 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->Lun0.IBase);
1165 PDMIBASE_RETURN_INTERFACE(pszIID, PDMISCSIPORT, &pThis->Lun0.IScsiPort);
1166 return NULL;
1167}
1168
1169
1170/**
1171 * Checks if all asynchronous I/O is finished.
1172 *
1173 * Used by usbMsdVMReset, usbMsdVMSuspend and usbMsdVMPowerOff.
1174 *
1175 * @returns true if quiesced, false if busy.
1176 * @param pUsbIns The USB device instance.
1177 */
1178static bool usbMsdAllAsyncIOIsFinished(PPDMUSBINS pUsbIns)
1179{
1180 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1181
1182 if ( VALID_PTR(pThis->pReq)
1183 && pThis->pReq->enmState == USBMSDREQSTATE_EXECUTING)
1184 return false;
1185
1186 return true;
1187}
1188
1189/**
1190 * @callback_method_impl{FNPDMDEVASYNCNOTIFY,
1191 * Callback employed by usbMsdVMSuspend and usbMsdVMPowerOff.}
1192 */
1193static DECLCALLBACK(bool) usbMsdIsAsyncSuspendOrPowerOffDone(PPDMUSBINS pUsbIns)
1194{
1195 if (!usbMsdAllAsyncIOIsFinished(pUsbIns))
1196 return false;
1197
1198 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1199 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
1200 return true;
1201}
1202
1203/**
1204 * Common worker for usbMsdVMSuspend and usbMsdVMPowerOff.
1205 */
1206static void usbMsdSuspendOrPowerOff(PPDMUSBINS pUsbIns)
1207{
1208 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1209
1210 ASMAtomicWriteBool(&pThis->fSignalIdle, true);
1211 if (!usbMsdAllAsyncIOIsFinished(pUsbIns))
1212 PDMUsbHlpSetAsyncNotification(pUsbIns, usbMsdIsAsyncSuspendOrPowerOffDone);
1213 else
1214 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
1215}
1216
1217
1218/* -=-=-=-=- Saved State -=-=-=-=- */
1219
1220/**
1221 * @copydoc FNUSBSSMSAVEPREP
1222 */
1223static DECLCALLBACK(int) usbMsdSavePrep(PPDMUSBINS pUsbIns, PSSMHANDLE pSSM)
1224{
1225 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1226
1227 Assert(usbMsdAllAsyncIOIsFinished(pUsbIns));
1228 Assert(usbMsdQueueIsEmpty(&pThis->ToHostQueue));
1229 Assert(usbMsdQueueIsEmpty(&pThis->DoneQueue));
1230 return VINF_SUCCESS;
1231}
1232
1233/**
1234 * @copydoc FNUSBSSMLOADPREP
1235 */
1236static DECLCALLBACK(int) usbMsdLoadPrep(PPDMUSBINS pUsbIns, PSSMHANDLE pSSM)
1237{
1238 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1239
1240 Assert(usbMsdAllAsyncIOIsFinished(pUsbIns));
1241 Assert(usbMsdQueueIsEmpty(&pThis->ToHostQueue));
1242 Assert(usbMsdQueueIsEmpty(&pThis->DoneQueue));
1243 return VINF_SUCCESS;
1244}
1245
1246/**
1247 * @copydoc FNUSBSSMLIVEEXEC
1248 */
1249static DECLCALLBACK(int) usbMsdLiveExec(PPDMUSBINS pUsbIns, PSSMHANDLE pSSM, uint32_t uPass)
1250{
1251 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1252
1253 /* config. */
1254 SSMR3PutBool(pSSM, pThis->Lun0.pIBase != NULL);
1255 return VINF_SSM_DONT_CALL_AGAIN;
1256}
1257
1258/**
1259 * @copydoc FNUSBSSMSAVEEXEC
1260 */
1261static DECLCALLBACK(int) usbMsdSaveExec(PPDMUSBINS pUsbIns, PSSMHANDLE pSSM)
1262{
1263 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1264 int rc;
1265
1266 /* The config */
1267 rc = usbMsdLiveExec(pUsbIns, pSSM, SSM_PASS_FINAL);
1268 AssertRCReturn(rc, rc);
1269
1270 SSMR3PutU8(pSSM, pThis->bConfigurationValue);
1271 SSMR3PutBool(pSSM, pThis->aEps[0].fHalted);
1272 SSMR3PutBool(pSSM, pThis->aEps[1].fHalted);
1273 SSMR3PutBool(pSSM, pThis->aEps[2].fHalted);
1274 SSMR3PutBool(pSSM, pThis->pReq != NULL);
1275
1276 if (pThis->pReq)
1277 {
1278 PUSBMSDREQ pReq = pThis->pReq;
1279
1280 SSMR3PutU32(pSSM, pReq->enmState);
1281 SSMR3PutU32(pSSM, pReq->cbBuf);
1282 if (pReq->cbBuf)
1283 {
1284 AssertPtr(pReq->pbBuf);
1285 SSMR3PutMem(pSSM, pReq->pbBuf, pReq->cbBuf);
1286 }
1287
1288 SSMR3PutU32(pSSM, pReq->offBuf);
1289 SSMR3PutMem(pSSM, &pReq->Cbw, sizeof(pReq->Cbw));
1290 SSMR3PutU32(pSSM, pReq->ScsiReq.uLogicalUnit);
1291 SSMR3PutU32(pSSM, pReq->ScsiReq.uDataDirection);
1292 SSMR3PutU32(pSSM, pReq->ScsiReq.cbCDB);
1293 SSMR3PutU32(pSSM, pReq->ScsiReq.cbScatterGather);
1294 SSMR3PutMem(pSSM, &pReq->ScsiReqSense[0], sizeof(pReq->ScsiReqSense));
1295 SSMR3PutS32(pSSM, pReq->iScsiReqStatus);
1296 }
1297
1298 return SSMR3PutU32(pSSM, UINT32_MAX); /* sanity/terminator */
1299}
1300
1301/**
1302 * @copydoc FNUSBSSMLOADEXEC
1303 */
1304static DECLCALLBACK(int) usbMsdLoadExec(PPDMUSBINS pUsbIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
1305{
1306 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1307 uint32_t u32;
1308 int rc;
1309
1310 if (uVersion > USB_MSD_SAVED_STATE_VERSION)
1311 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
1312
1313 /* Verify config. */
1314 bool fInUse;
1315 rc = SSMR3GetBool(pSSM, &fInUse);
1316 AssertRCReturn(rc, rc);
1317 if (fInUse != (pThis->Lun0.pIBase != NULL))
1318 return SSMR3SetCfgError(pSSM, RT_SRC_POS,
1319 N_("The %s VM is missing a USB mass storage device. Please make sure the source and target VMs have compatible storage configurations"),
1320 fInUse ? "target" : "source");
1321
1322 if (uPass == SSM_PASS_FINAL)
1323 {
1324 /* Restore data. */
1325 bool fReqAlloc = false;
1326
1327 Assert(!pThis->pReq);
1328
1329 SSMR3GetU8(pSSM, &pThis->bConfigurationValue);
1330 SSMR3GetBool(pSSM, &pThis->aEps[0].fHalted);
1331 SSMR3GetBool(pSSM, &pThis->aEps[1].fHalted);
1332 SSMR3GetBool(pSSM, &pThis->aEps[2].fHalted);
1333 SSMR3GetBool(pSSM, &fReqAlloc);
1334
1335 if (fReqAlloc)
1336 {
1337 PUSBMSDREQ pReq = usbMsdReqAlloc(pUsbIns);
1338
1339 if (pReq)
1340 {
1341 uint32_t cbBuf = 0;
1342
1343 pThis->pReq = pReq;
1344
1345 SSMR3GetU32(pSSM, (uint32_t *)&pReq->enmState);
1346 SSMR3GetU32(pSSM, &cbBuf);
1347 if (cbBuf)
1348 {
1349 if (usbMsdReqEnsureBuffer(pReq, cbBuf))
1350 {
1351 AssertPtr(pReq->pbBuf);
1352 Assert(cbBuf == pReq->cbBuf);
1353 SSMR3GetMem(pSSM, pReq->pbBuf, pReq->cbBuf);
1354 }
1355 else
1356 rc = VERR_NO_MEMORY;
1357 }
1358
1359 if (RT_SUCCESS(rc))
1360 {
1361 SSMR3GetU32(pSSM, &pReq->offBuf);
1362 SSMR3GetMem(pSSM, &pReq->Cbw, sizeof(pReq->Cbw));
1363 SSMR3GetU32(pSSM, &pReq->ScsiReq.uLogicalUnit);
1364 SSMR3GetU32(pSSM, (uint32_t *)&pReq->ScsiReq.uDataDirection);
1365 SSMR3GetU32(pSSM, &pReq->ScsiReq.cbCDB);
1366 SSMR3GetU32(pSSM, &pReq->ScsiReq.cbScatterGather);
1367 SSMR3GetMem(pSSM, &pReq->ScsiReqSense[0], sizeof(pReq->ScsiReqSense));
1368 SSMR3GetS32(pSSM, &pReq->iScsiReqStatus);
1369
1370 /* Setup the rest of the SCSI request. */
1371 pReq->ScsiReq.cbCDB = pReq->Cbw.bCBWCBLength;
1372 pReq->ScsiReq.pbCDB = &pReq->Cbw.CBWCB[0];
1373 pReq->ScsiReqSeg.pvSeg = pReq->pbBuf;
1374 pReq->ScsiReqSeg.cbSeg = pReq->ScsiReq.cbScatterGather;
1375 pReq->ScsiReq.cScatterGatherEntries = 1;
1376 pReq->ScsiReq.paScatterGatherHead = &pReq->ScsiReqSeg;
1377 pReq->ScsiReq.cbSenseBuffer = sizeof(pReq->ScsiReqSense);
1378 pReq->ScsiReq.pbSenseBuffer = &pReq->ScsiReqSense[0];
1379 pReq->ScsiReq.pvUser = NULL;
1380 }
1381 }
1382 else
1383 rc = VERR_NO_MEMORY;
1384 }
1385
1386 if (RT_SUCCESS(rc))
1387 rc = SSMR3GetU32(pSSM, &u32);
1388
1389 if (RT_FAILURE(rc))
1390 return rc;
1391 AssertMsgReturn(u32 == UINT32_MAX, ("%#x\n", u32), VERR_SSM_DATA_UNIT_FORMAT_CHANGED);
1392 }
1393
1394 return VINF_SUCCESS;
1395}
1396
1397
1398/**
1399 * @copydoc PDMUSBREG::pfnUrbReap
1400 */
1401static DECLCALLBACK(PVUSBURB) usbMsdUrbReap(PPDMUSBINS pUsbIns, RTMSINTERVAL cMillies)
1402{
1403 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1404 LogFlow(("usbMsdUrbReap/#%u: cMillies=%u\n", pUsbIns->iInstance, cMillies));
1405
1406 RTCritSectEnter(&pThis->CritSect);
1407
1408 PVUSBURB pUrb = usbMsdQueueRemoveHead(&pThis->DoneQueue);
1409 if (!pUrb && cMillies)
1410 {
1411 /* Wait */
1412 pThis->fHaveDoneQueueWaiter = true;
1413 RTCritSectLeave(&pThis->CritSect);
1414
1415 RTSemEventWait(pThis->hEvtDoneQueue, cMillies);
1416
1417 RTCritSectEnter(&pThis->CritSect);
1418 pThis->fHaveDoneQueueWaiter = false;
1419
1420 pUrb = usbMsdQueueRemoveHead(&pThis->DoneQueue);
1421 }
1422
1423 RTCritSectLeave(&pThis->CritSect);
1424
1425 if (pUrb)
1426 Log(("usbMsdUrbReap/#%u: pUrb=%p:%s\n", pUsbIns->iInstance, pUrb, pUrb->pszDesc));
1427 return pUrb;
1428}
1429
1430
1431/**
1432 * @copydoc PDMUSBREG::pfnWakeup
1433 */
1434static DECLCALLBACK(int) usbMsdWakeup(PPDMUSBINS pUsbIns)
1435{
1436 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1437 LogFlow(("usbMsdUrbReap/#%u:\n", pUsbIns->iInstance));
1438
1439 return RTSemEventSignal(pThis->hEvtDoneQueue);
1440}
1441
1442
1443/**
1444 * @copydoc PDMUSBREG::pfnUrbCancel
1445 */
1446static DECLCALLBACK(int) usbMsdUrbCancel(PPDMUSBINS pUsbIns, PVUSBURB pUrb)
1447{
1448 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1449 LogFlow(("usbMsdUrbCancel/#%u: pUrb=%p:%s\n", pUsbIns->iInstance, pUrb, pUrb->pszDesc));
1450 RTCritSectEnter(&pThis->CritSect);
1451
1452 /*
1453 * Remove the URB from the to-host queue and move it onto the done queue.
1454 */
1455 if (usbMsdQueueRemove(&pThis->ToHostQueue, pUrb))
1456 usbMsdLinkDone(pThis, pUrb);
1457
1458 RTCritSectLeave(&pThis->CritSect);
1459 return VINF_SUCCESS;
1460}
1461
1462
1463/**
1464 * Fails an illegal SCSI request.
1465 *
1466 * @returns VBox status code.
1467 * @param pThis The MSD instance data.
1468 * @param pReq The MSD request.
1469 * @param bAsc The ASC for the SCSI_SENSE_ILLEGAL_REQUEST.
1470 * @param bAscq The ASC qualifier.
1471 * @param pszWhy For logging why.
1472 */
1473static int usbMsdScsiIllegalRequest(PUSBMSD pThis, PUSBMSDREQ pReq, uint8_t bAsc, uint8_t bAscq, const char *pszWhy)
1474{
1475 Log(("usbMsdScsiIllegalRequest: bAsc=%#x bAscq=%#x %s\n", bAsc, bAscq, pszWhy));
1476
1477 RT_ZERO(pReq->ScsiReqSense);
1478 pReq->ScsiReqSense[0] = 0x80 | SCSI_SENSE_RESPONSE_CODE_CURR_FIXED;
1479 pReq->ScsiReqSense[2] = SCSI_SENSE_ILLEGAL_REQUEST;
1480 pReq->ScsiReqSense[7] = 10;
1481 pReq->ScsiReqSense[12] = SCSI_ASC_INVALID_MESSAGE;
1482 pReq->ScsiReqSense[13] = 0; /* Should be ASCQ but it has the same value for success. */
1483
1484 usbMsdLun0ScsiRequestCompleted(&pThis->Lun0.IScsiPort, &pReq->ScsiReq, SCSI_STATUS_CHECK_CONDITION, false, VINF_SUCCESS);
1485 return VINF_SUCCESS;
1486}
1487
1488
1489/**
1490 * The SCSI driver doesn't handle SCSI_REQUEST_SENSE but instead
1491 * returns the sense info with the request.
1492 *
1493 */
1494static int usbMsdHandleScsiReqestSense(PUSBMSD pThis, PUSBMSDREQ pReq, PCUSBCBW pCbw)
1495{
1496 Log(("usbMsdHandleScsiReqestSense: Entering EXECUTING (dCBWTag=%#x).\n", pReq->Cbw.dCBWTag));
1497 Assert(pReq == pThis->pReq);
1498 pReq->enmState = USBMSDREQSTATE_EXECUTING;
1499
1500 /* validation */
1501 if ((pCbw->bmCBWFlags & USBCBW_DIR_MASK) != USBCBW_DIR_IN)
1502 return usbMsdScsiIllegalRequest(pThis, pReq, SCSI_ASC_INVALID_MESSAGE, 0, "direction");
1503 if (pCbw->bCBWCBLength < 6)
1504 return usbMsdScsiIllegalRequest(pThis, pReq, SCSI_ASC_INVALID_MESSAGE, 0, "length");
1505 if ((pCbw->CBWCB[1] >> 5) != pCbw->bCBWLun)
1506 return usbMsdScsiIllegalRequest(pThis, pReq, SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0, "lun");
1507 if (pCbw->bCBWLun != 0)
1508 return usbMsdScsiIllegalRequest(pThis, pReq, SCSI_ASC_INVALID_MESSAGE, 0, "lun0");
1509 if (pCbw->CBWCB[4] < 6)
1510 return usbMsdScsiIllegalRequest(pThis, pReq, SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0, "out length");
1511
1512 /* If the previous command succeeded successfully, whip up some sense data. */
1513 if ( pReq->iScsiReqStatus == SCSI_STATUS_OK
1514 && pReq->ScsiReqSense[0] == 0)
1515 {
1516 RT_ZERO(pReq->ScsiReqSense);
1517#if 0 /** @todo something upsets linux about this stuff. Needs investigation. */
1518 pReq->ScsiReqSense[0] = 0x80 | SCSI_SENSE_RESPONSE_CODE_CURR_FIXED;
1519 pReq->ScsiReqSense[0] = SCSI_SENSE_RESPONSE_CODE_CURR_FIXED;
1520 pReq->ScsiReqSense[2] = SCSI_SENSE_NONE;
1521 pReq->ScsiReqSense[7] = 10;
1522 pReq->ScsiReqSense[12] = SCSI_ASC_NONE;
1523 pReq->ScsiReqSense[13] = SCSI_ASC_NONE; /* Should be ASCQ but it has the same value for success. */
1524#endif
1525 }
1526
1527 /* Copy the data into the result buffer. */
1528 size_t cbCopy = RT_MIN(pCbw->dCBWDataTransferLength, sizeof(pReq->ScsiReqSense));
1529 Log(("usbMsd: SCSI_REQUEST_SENSE - CBWCB[4]=%#x iOldState=%d, %u bytes, raw: %.*Rhxs\n",
1530 pCbw->CBWCB[4], pReq->iScsiReqStatus, pCbw->dCBWDataTransferLength, RT_MAX(1, cbCopy), pReq->ScsiReqSense));
1531 memcpy(pReq->pbBuf, &pReq->ScsiReqSense[0], cbCopy);
1532
1533 usbMsdReqPrepare(pReq, pCbw);
1534
1535 /* Do normal completion. */
1536 usbMsdLun0ScsiRequestCompleted(&pThis->Lun0.IScsiPort, &pReq->ScsiReq, SCSI_STATUS_OK, false, VINF_SUCCESS);
1537 return VINF_SUCCESS;
1538}
1539
1540
1541/**
1542 * Wrapper around PDMISCSICONNECTOR::pfnSCSIRequestSend that deals with
1543 * SCSI_REQUEST_SENSE.
1544 *
1545 * @returns VBox status code.
1546 * @param pThis The MSD instance data.
1547 * @param pReq The MSD request.
1548 * @param pszCaller Where we're called from.
1549 */
1550static int usbMsdSubmitScsiCommand(PUSBMSD pThis, PUSBMSDREQ pReq, const char *pszCaller)
1551{
1552 Log(("%s: Entering EXECUTING (dCBWTag=%#x).\n", pszCaller, pReq->Cbw.dCBWTag));
1553 Assert(pReq == pThis->pReq);
1554 pReq->enmState = USBMSDREQSTATE_EXECUTING;
1555
1556 switch (pReq->ScsiReq.pbCDB[0])
1557 {
1558 case SCSI_REQUEST_SENSE:
1559 {
1560 }
1561
1562 default:
1563 return pThis->Lun0.pIScsiConnector->pfnSCSIRequestSend(pThis->Lun0.pIScsiConnector, &pReq->ScsiReq);
1564 }
1565}
1566
1567/**
1568 * Validates a SCSI request before passing it down to the SCSI driver.
1569 *
1570 * @returns true / false. The request will be completed on failure.
1571 * @param pThis The MSD instance data.
1572 * @param pCbw The USB command block wrapper.
1573 * @param pUrb The URB.
1574 */
1575static bool usbMsdIsValidCommand(PUSBMSD pThis, PCUSBCBW pCbw, PVUSBURB pUrb)
1576{
1577 switch (pCbw->CBWCB[0])
1578 {
1579 case SCSI_REQUEST_SENSE:
1580 /** @todo validate this. */
1581 return true;
1582
1583 default:
1584 return true;
1585 }
1586}
1587
1588
1589/**
1590 * Handle requests sent to the outbound (to device) bulk pipe.
1591 */
1592static int usbMsdHandleBulkHostToDev(PUSBMSD pThis, PUSBMSDEP pEp, PVUSBURB pUrb)
1593{
1594 /*
1595 * Stall the request if the pipe is halted.
1596 */
1597 if (RT_UNLIKELY(pEp->fHalted))
1598 return usbMsdCompleteStall(pThis, NULL, pUrb, "Halted pipe");
1599
1600 /*
1601 * Deal with the URB according to the current state.
1602 */
1603 PUSBMSDREQ pReq = pThis->pReq;
1604 USBMSDREQSTATE enmState = pReq ? pReq->enmState : USBMSDREQSTATE_READY;
1605 switch (enmState)
1606 {
1607 case USBMSDREQSTATE_STATUS:
1608 LogFlow(("usbMsdHandleBulkHostToDev: Skipping pending status.\n"));
1609 pReq->enmState = USBMSDREQSTATE_READY;
1610 /* fall thru */
1611
1612 /*
1613 * We're ready to receive a command. Start off by validating the
1614 * incoming request.
1615 */
1616 case USBMSDREQSTATE_READY:
1617 {
1618 PCUSBCBW pCbw = (PUSBCBW)&pUrb->abData[0];
1619 if (pUrb->cbData < RT_UOFFSETOF(USBCBW, CBWCB[1]))
1620 {
1621 Log(("usbMsd: Bad CBW: cbData=%#x < min=%#x\n", pUrb->cbData, RT_UOFFSETOF(USBCBW, CBWCB[1]) ));
1622 return usbMsdCompleteStall(pThis, NULL, pUrb, "BAD CBW");
1623 }
1624 if (pCbw->dCBWSignature != USBCBW_SIGNATURE)
1625 {
1626 Log(("usbMsd: CBW: Invalid dCBWSignature value: %#x\n", pCbw->dCBWSignature));
1627 return usbMsdCompleteStall(pThis, NULL, pUrb, "Bad CBW");
1628 }
1629 Log(("usbMsd: CBW: dCBWTag=%#x dCBWDataTransferLength=%#x bmCBWFlags=%#x bCBWLun=%#x bCBWCBLength=%#x cbData=%#x fShortNotOk=%RTbool\n",
1630 pCbw->dCBWTag, pCbw->dCBWDataTransferLength, pCbw->bmCBWFlags, pCbw->bCBWLun, pCbw->bCBWCBLength, pUrb->cbData, pUrb->fShortNotOk));
1631 if (pCbw->bmCBWFlags & ~USBCBW_DIR_MASK)
1632 {
1633 Log(("usbMsd: CBW: Bad bmCBWFlags value: %#x\n", pCbw->bmCBWFlags));
1634 return usbMsdCompleteStall(pThis, NULL, pUrb, "Bad CBW");
1635
1636 }
1637 if (pCbw->bCBWLun != 0)
1638 {
1639 Log(("usbMsd: CBW: Bad bCBWLun value: %#x\n", pCbw->bCBWLun));
1640 return usbMsdCompleteStall(pThis, NULL, pUrb, "Bad CBW");
1641 }
1642 if (pCbw->bCBWCBLength == 0)
1643 {
1644 Log(("usbMsd: CBW: Bad bCBWCBLength value: %#x\n", pCbw->bCBWCBLength));
1645 return usbMsdCompleteStall(pThis, NULL, pUrb, "Bad CBW");
1646 }
1647 if (pUrb->cbData < RT_UOFFSETOF(USBCBW, CBWCB[pCbw->bCBWCBLength]))
1648 {
1649 Log(("usbMsd: CBW: Mismatching cbData and bCBWCBLength values: %#x vs. %#x (%#x)\n",
1650 pUrb->cbData, RT_UOFFSETOF(USBCBW, CBWCB[pCbw->bCBWCBLength]), pCbw->bCBWCBLength));
1651 return usbMsdCompleteStall(pThis, NULL, pUrb, "Bad CBW");
1652 }
1653 if (pCbw->dCBWDataTransferLength > _1M)
1654 {
1655 Log(("usbMsd: CBW: dCBWDataTransferLength is too large: %#x (%u)\n",
1656 pCbw->dCBWDataTransferLength, pCbw->dCBWDataTransferLength));
1657 return usbMsdCompleteStall(pThis, NULL, pUrb, "Too big transfer");
1658 }
1659
1660 if (!usbMsdIsValidCommand(pThis, pCbw, pUrb))
1661 return VINF_SUCCESS;
1662
1663 /*
1664 * Make sure we've got a request and a sufficient buffer space.
1665 *
1666 * Note! This will make sure the buffer is ZERO as well, thus
1667 * saving us the trouble of clearing the output buffer on
1668 * failure later.
1669 */
1670 if (!pReq)
1671 {
1672 pReq = usbMsdReqAlloc(pThis->pUsbIns);
1673 if (!pReq)
1674 return usbMsdCompleteStall(pThis, NULL, pUrb, "Request allocation failure");
1675 pThis->pReq = pReq;
1676 }
1677 if (!usbMsdReqEnsureBuffer(pReq, pCbw->dCBWDataTransferLength))
1678 return usbMsdCompleteStall(pThis, NULL, pUrb, "Buffer allocation failure");
1679
1680 /*
1681 * Special case REQUEST SENSE requests, usbMsdReqPrepare will
1682 * trash the sense data otherwise.
1683 */
1684 if (pCbw->CBWCB[0] == SCSI_REQUEST_SENSE)
1685 usbMsdHandleScsiReqestSense(pThis, pReq, pCbw);
1686 else
1687 {
1688 /*
1689 * Prepare the request. Kick it off right away if possible.
1690 */
1691 usbMsdReqPrepare(pReq, pCbw);
1692
1693 if ( pReq->Cbw.dCBWDataTransferLength == 0
1694 || (pReq->Cbw.bmCBWFlags & USBCBW_DIR_MASK) == USBCBW_DIR_IN)
1695 {
1696 int rc = usbMsdSubmitScsiCommand(pThis, pReq, "usbMsdHandleBulkHostToDev");
1697 if (RT_FAILURE(rc))
1698 {
1699 Log(("usbMsd: Failed sending SCSI request to driver: %Rrc\n", rc));
1700 return usbMsdCompleteStall(pThis, NULL, pUrb, "SCSI Submit #1");
1701 }
1702 }
1703 else
1704 {
1705 Log(("usbMsdHandleBulkHostToDev: Entering DATA_FROM_HOST.\n"));
1706 pReq->enmState = USBMSDREQSTATE_DATA_FROM_HOST;
1707 }
1708 }
1709
1710 return usbMsdCompleteOk(pThis, pUrb, pUrb->cbData);
1711 }
1712
1713 /*
1714 * Stuff the data into the buffer.
1715 */
1716 case USBMSDREQSTATE_DATA_FROM_HOST:
1717 {
1718 uint32_t cbData = pUrb->cbData;
1719 uint32_t cbLeft = pReq->Cbw.dCBWDataTransferLength - pReq->offBuf;
1720 if (cbData > cbLeft)
1721 {
1722 Log(("usbMsd: Too much data: cbData=%#x offBuf=%#x dCBWDataTransferLength=%#x cbLeft=%#x\n",
1723 cbData, pReq->offBuf, pReq->Cbw.dCBWDataTransferLength, cbLeft));
1724 return usbMsdCompleteStall(pThis, NULL, pUrb, "Too much data");
1725 }
1726 memcpy(&pReq->pbBuf[pReq->offBuf], &pUrb->abData[0], cbData);
1727 pReq->offBuf += cbData;
1728
1729 if (pReq->offBuf == pReq->Cbw.dCBWDataTransferLength)
1730 {
1731 int rc = usbMsdSubmitScsiCommand(pThis, pReq, "usbMsdHandleBulkHostToDev");
1732 if (RT_FAILURE(rc))
1733 {
1734 Log(("usbMsd: Failed sending SCSI request to driver: %Rrc\n", rc));
1735 return usbMsdCompleteStall(pThis, NULL, pUrb, "SCSI Submit #2");
1736 }
1737 }
1738 return usbMsdCompleteOk(pThis, pUrb, cbData);
1739 }
1740
1741 /*
1742 * Bad state, stall.
1743 */
1744 case USBMSDREQSTATE_DATA_TO_HOST:
1745 return usbMsdCompleteStall(pThis, NULL, pUrb, "Bad state H2D: DATA_TO_HOST");
1746
1747 case USBMSDREQSTATE_EXECUTING:
1748 return usbMsdCompleteStall(pThis, NULL, pUrb, "Bad state H2D: EXECUTING");
1749
1750 default:
1751 AssertMsgFailed(("enmState=%d\n", enmState));
1752 return usbMsdCompleteStall(pThis, NULL, pUrb, "Bad state (H2D)");
1753 }
1754}
1755
1756
1757/**
1758 * Handle requests sent to the inbound (to host) bulk pipe.
1759 */
1760static int usbMsdHandleBulkDevToHost(PUSBMSD pThis, PUSBMSDEP pEp, PVUSBURB pUrb)
1761{
1762 /*
1763 * Stall the request if the pipe is halted OR if there is no
1764 * pending request yet.
1765 */
1766 PUSBMSDREQ pReq = pThis->pReq;
1767 if (RT_UNLIKELY(pEp->fHalted || !pReq))
1768 return usbMsdCompleteStall(pThis, NULL, pUrb, pEp->fHalted ? "Halted pipe" : "No request");
1769
1770 /*
1771 * Deal with the URB according to the state.
1772 */
1773 switch (pReq->enmState)
1774 {
1775 /*
1776 * We've data left to transfer to the host.
1777 */
1778 case USBMSDREQSTATE_DATA_TO_HOST:
1779 {
1780 uint32_t cbData = pUrb->cbData;
1781 uint32_t cbCopy = pReq->Cbw.dCBWDataTransferLength - pReq->offBuf;
1782 if (cbData <= cbCopy)
1783 cbCopy = cbData;
1784 else if (pUrb->fShortNotOk)
1785 {
1786 Log(("usbMsd: Requested more data that we've got; cbData=%#x offBuf=%#x dCBWDataTransferLength=%#x cbLeft=%#x\n",
1787 cbData, pReq->offBuf, pReq->Cbw.dCBWDataTransferLength, cbCopy));
1788 return usbMsdCompleteStall(pThis, NULL, pUrb, "Data underrun");
1789 }
1790 memcpy(&pUrb->abData[0], &pReq->pbBuf[pReq->offBuf], cbCopy);
1791 pReq->offBuf += cbCopy;
1792
1793 if (pReq->offBuf == pReq->Cbw.dCBWDataTransferLength)
1794 {
1795 Log(("usbMsdHandleBulkDevToHost: Entering STATUS\n"));
1796 pReq->enmState = USBMSDREQSTATE_STATUS;
1797 }
1798 return usbMsdCompleteOk(pThis, pUrb, cbCopy);
1799 }
1800
1801 /*
1802 * Status transfer.
1803 */
1804 case USBMSDREQSTATE_STATUS:
1805 {
1806 if ((pUrb->cbData < sizeof(USBCSW)) || (pUrb->cbData > sizeof(USBCSW) && pUrb->fShortNotOk))
1807 {
1808 Log(("usbMsd: Unexpected status request size: %#x (expected %#x), fShortNotOK=%RTbool\n", pUrb->cbData, sizeof(USBCSW), pUrb->fShortNotOk));
1809 return usbMsdCompleteStall(pThis, NULL, pUrb, "Invalid CSW size");
1810 }
1811
1812 /* Enter a CSW into the URB data buffer. */
1813 PUSBCSW pCsw = (PUSBCSW)&pUrb->abData[0];
1814 pCsw->dCSWSignature = USBCSW_SIGNATURE;
1815 pCsw->dCSWTag = pReq->Cbw.dCBWTag;
1816 pCsw->bCSWStatus = pReq->iScsiReqStatus == SCSI_STATUS_OK
1817 ? USBCSW_STATUS_OK
1818 : pReq->iScsiReqStatus >= 0
1819 ? USBCSW_STATUS_FAILED
1820 : USBCSW_STATUS_PHASE_ERROR;
1821 /** @todo the following is not always accurate; VSCSI needs
1822 * to implement residual counts properly! */
1823 if ((pReq->Cbw.bmCBWFlags & USBCBW_DIR_MASK) == USBCBW_DIR_OUT)
1824 pCsw->dCSWDataResidue = pCsw->bCSWStatus == USBCSW_STATUS_OK
1825 ? pReq->Cbw.dCBWDataTransferLength - pReq->ScsiReq.cbScatterGather
1826 : pReq->Cbw.dCBWDataTransferLength;
1827 else
1828 pCsw->dCSWDataResidue = pCsw->bCSWStatus == USBCSW_STATUS_OK
1829 ? 0
1830 : pReq->ScsiReq.cbScatterGather;
1831 Log(("usbMsd: CSW: dCSWTag=%#x bCSWStatus=%d dCSWDataResidue=%#x\n",
1832 pCsw->dCSWTag, pCsw->bCSWStatus, pCsw->dCSWDataResidue));
1833
1834 Log(("usbMsdHandleBulkDevToHost: Entering READY\n"));
1835 pReq->enmState = USBMSDREQSTATE_READY;
1836 return usbMsdCompleteOk(pThis, pUrb, sizeof(*pCsw));
1837 }
1838
1839 /*
1840 * Status request before we've received all (or even any) data.
1841 * Linux 2.4.31 does this sometimes. The recommended behavior is to
1842 * to accept the current data amount and execute the request. (The
1843 * alternative behavior is to stall.)
1844 */
1845 case USBMSDREQSTATE_DATA_FROM_HOST:
1846 {
1847 if (pUrb->cbData != sizeof(USBCSW))
1848 {
1849 Log(("usbMsdHandleBulkDevToHost: DATA_FROM_HOST; cbData=%#x -> stall\n", pUrb->cbData));
1850 return usbMsdCompleteStall(pThis, NULL, pUrb, "Invalid CSW size");
1851 }
1852
1853 /* Adjust the request and kick it off. Special case the no-data
1854 case since the SCSI driver doesn't like that. */
1855 pReq->ScsiReq.cbScatterGather = pReq->offBuf;
1856 pReq->ScsiReqSeg.cbSeg = pReq->offBuf;
1857 if (!pReq->offBuf)
1858 {
1859 Log(("usbMsdHandleBulkDevToHost: Entering EXECUTING (offBuf=0x0).\n"));
1860 pReq->enmState = USBMSDREQSTATE_EXECUTING;
1861
1862 usbMsdQueueAddTail(&pThis->ToHostQueue, pUrb);
1863 LogFlow(("usbMsdHandleBulkDevToHost: Added %p:%s to the to-host queue\n", pUrb, pUrb->pszDesc));
1864
1865 usbMsdLun0ScsiRequestCompleted(&pThis->Lun0.IScsiPort, &pReq->ScsiReq, SCSI_STATUS_OK, false, VINF_SUCCESS);
1866 return VINF_SUCCESS;
1867 }
1868
1869 int rc = usbMsdSubmitScsiCommand(pThis, pReq, "usbMsdHandleBulkDevToHost");
1870 if (RT_FAILURE(rc))
1871 {
1872 Log(("usbMsd: Failed sending SCSI request to driver: %Rrc\n", rc));
1873 return usbMsdCompleteStall(pThis, NULL, pUrb, "SCSI Submit #3");
1874 }
1875
1876 /* fall thru */
1877 }
1878
1879 /*
1880 * The SCSI command is still pending, queue the URB awaiting its
1881 * completion.
1882 */
1883 case USBMSDREQSTATE_EXECUTING:
1884 usbMsdQueueAddTail(&pThis->ToHostQueue, pUrb);
1885 LogFlow(("usbMsdHandleBulkDevToHost: Added %p:%s to the to-host queue\n", pUrb, pUrb->pszDesc));
1886 return VINF_SUCCESS;
1887
1888 /*
1889 * Bad states, stall.
1890 */
1891 case USBMSDREQSTATE_READY:
1892 Log(("usbMsdHandleBulkDevToHost: enmState=READ(%d) (cbData=%#x)\n", pReq->enmState, pUrb->cbData));
1893 return usbMsdCompleteStall(pThis, NULL, pUrb, "Bad state D2H: READY");
1894
1895 default:
1896 Log(("usbMsdHandleBulkDevToHost: enmState=%d cbData=%#x\n", pReq->enmState, pUrb->cbData));
1897 return usbMsdCompleteStall(pThis, NULL, pUrb, "Really bad state (D2H)!");
1898 }
1899}
1900
1901
1902/**
1903 * Handles request send to the default control pipe.
1904 */
1905static int usbMsdHandleDefaultPipe(PUSBMSD pThis, PUSBMSDEP pEp, PVUSBURB pUrb)
1906{
1907 PVUSBSETUP pSetup = (PVUSBSETUP)&pUrb->abData[0];
1908 AssertReturn(pUrb->cbData >= sizeof(*pSetup), VERR_VUSB_FAILED_TO_QUEUE_URB);
1909
1910 if ((pSetup->bmRequestType & VUSB_REQ_MASK) == VUSB_REQ_STANDARD)
1911 {
1912 switch (pSetup->bRequest)
1913 {
1914 case VUSB_REQ_GET_DESCRIPTOR:
1915 {
1916 if (pSetup->bmRequestType != (VUSB_TO_DEVICE | VUSB_REQ_STANDARD | VUSB_DIR_TO_HOST))
1917 {
1918 Log(("usbMsd: Bad GET_DESCRIPTOR req: bmRequestType=%#x\n", pSetup->bmRequestType));
1919 return usbMsdCompleteStall(pThis, pEp, pUrb, "Bad GET_DESCRIPTOR");
1920 }
1921
1922 switch (pSetup->wValue >> 8)
1923 {
1924 uint32_t cbCopy;
1925
1926 case VUSB_DT_STRING:
1927 Log(("usbMsd: GET_DESCRIPTOR DT_STRING wValue=%#x wIndex=%#x\n", pSetup->wValue, pSetup->wIndex));
1928 break;
1929 case VUSB_DT_DEVICE_QUALIFIER:
1930 Log(("usbMsd: GET_DESCRIPTOR DT_DEVICE_QUALIFIER wValue=%#x wIndex=%#x\n", pSetup->wValue, pSetup->wIndex));
1931 /* Returned data is written after the setup message. */
1932 cbCopy = pUrb->cbData - sizeof(*pSetup);
1933 cbCopy = RT_MIN(cbCopy, sizeof(g_UsbMsdDeviceQualifier));
1934 memcpy(&pUrb->abData[sizeof(*pSetup)], &g_UsbMsdDeviceQualifier, cbCopy);
1935 return usbMsdCompleteOk(pThis, pUrb, cbCopy + sizeof(*pSetup));
1936 case VUSB_DT_BOS:
1937 Log(("usbMsd: GET_DESCRIPTOR DT_BOS wValue=%#x wIndex=%#x\n", pSetup->wValue, pSetup->wIndex));
1938 /* Returned data is written after the setup message. */
1939 cbCopy = pUrb->cbData - sizeof(*pSetup);
1940 cbCopy = RT_MIN(cbCopy, sizeof(g_UsbMsdBOS));
1941 memcpy(&pUrb->abData[sizeof(*pSetup)], &g_UsbMsdBOS, cbCopy);
1942 return usbMsdCompleteOk(pThis, pUrb, cbCopy + sizeof(*pSetup));
1943 default:
1944 Log(("usbMsd: GET_DESCRIPTOR, huh? wValue=%#x wIndex=%#x\n", pSetup->wValue, pSetup->wIndex));
1945 break;
1946 }
1947 break;
1948 }
1949
1950 case VUSB_REQ_CLEAR_FEATURE:
1951 break;
1952 }
1953
1954 /** @todo implement this. */
1955 Log(("usbMsd: Implement standard request: bmRequestType=%#x bRequest=%#x wValue=%#x wIndex=%#x wLength=%#x\n",
1956 pSetup->bmRequestType, pSetup->bRequest, pSetup->wValue, pSetup->wIndex, pSetup->wLength));
1957
1958 usbMsdCompleteStall(pThis, pEp, pUrb, "TODO: standard request stuff");
1959 }
1960 /* 3.1 Bulk-Only Mass Storage Reset */
1961 else if ( pSetup->bmRequestType == (VUSB_REQ_CLASS | VUSB_TO_INTERFACE)
1962 && pSetup->bRequest == 0xff
1963 && !pSetup->wValue
1964 && !pSetup->wLength
1965 && pSetup->wIndex == 0)
1966 {
1967 Log(("usbMsdHandleDefaultPipe: Bulk-Only Mass Storage Reset\n"));
1968 return usbMsdResetWorker(pThis, pUrb, false /*fSetConfig*/);
1969 }
1970 /* 3.2 Get Max LUN, may stall if we like (but we don't). */
1971 else if ( pSetup->bmRequestType == (VUSB_REQ_CLASS | VUSB_TO_INTERFACE | VUSB_DIR_TO_HOST)
1972 && pSetup->bRequest == 0xfe
1973 && !pSetup->wValue
1974 && pSetup->wLength == 1
1975 && pSetup->wIndex == 0)
1976 {
1977 *(uint8_t *)(pSetup + 1) = 0; /* max lun is 0 */
1978 usbMsdCompleteOk(pThis, pUrb, 1);
1979 }
1980 else
1981 {
1982 Log(("usbMsd: Unknown control msg: bmRequestType=%#x bRequest=%#x wValue=%#x wIndex=%#x wLength=%#x\n",
1983 pSetup->bmRequestType, pSetup->bRequest, pSetup->wValue, pSetup->wIndex, pSetup->wLength));
1984 return usbMsdCompleteStall(pThis, pEp, pUrb, "Unknown control msg");
1985 }
1986
1987 return VINF_SUCCESS;
1988}
1989
1990
1991/**
1992 * @copydoc PDMUSBREG::pfnQueue
1993 */
1994static DECLCALLBACK(int) usbMsdQueue(PPDMUSBINS pUsbIns, PVUSBURB pUrb)
1995{
1996 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
1997 LogFlow(("usbMsdQueue/#%u: pUrb=%p:%s EndPt=%#x\n", pUsbIns->iInstance, pUrb, pUrb->pszDesc, pUrb->EndPt));
1998 RTCritSectEnter(&pThis->CritSect);
1999
2000 /*
2001 * Parse on a per end-point basis.
2002 */
2003 int rc;
2004 switch (pUrb->EndPt)
2005 {
2006 case 0:
2007 rc = usbMsdHandleDefaultPipe(pThis, &pThis->aEps[0], pUrb);
2008 break;
2009
2010 case 0x81:
2011 AssertFailed();
2012 case 0x01:
2013 rc = usbMsdHandleBulkDevToHost(pThis, &pThis->aEps[1], pUrb);
2014 break;
2015
2016 case 0x02:
2017 rc = usbMsdHandleBulkHostToDev(pThis, &pThis->aEps[2], pUrb);
2018 break;
2019
2020 default:
2021 AssertMsgFailed(("EndPt=%d\n", pUrb->EndPt));
2022 rc = VERR_VUSB_FAILED_TO_QUEUE_URB;
2023 break;
2024 }
2025
2026 RTCritSectLeave(&pThis->CritSect);
2027 return rc;
2028}
2029
2030
2031/**
2032 * @copydoc PDMUSBREG::pfnUsbClearHaltedEndpoint
2033 */
2034static DECLCALLBACK(int) usbMsdUsbClearHaltedEndpoint(PPDMUSBINS pUsbIns, unsigned uEndpoint)
2035{
2036 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2037 LogFlow(("usbMsdUsbClearHaltedEndpoint/#%u: uEndpoint=%#x\n", pUsbIns->iInstance, uEndpoint));
2038
2039 if ((uEndpoint & ~0x80) < RT_ELEMENTS(pThis->aEps))
2040 {
2041 RTCritSectEnter(&pThis->CritSect);
2042 pThis->aEps[(uEndpoint & ~0x80)].fHalted = false;
2043 RTCritSectLeave(&pThis->CritSect);
2044 }
2045
2046 return VINF_SUCCESS;
2047}
2048
2049
2050/**
2051 * @copydoc PDMUSBREG::pfnUsbSetInterface
2052 */
2053static DECLCALLBACK(int) usbMsdUsbSetInterface(PPDMUSBINS pUsbIns, uint8_t bInterfaceNumber, uint8_t bAlternateSetting)
2054{
2055 LogFlow(("usbMsdUsbSetInterface/#%u: bInterfaceNumber=%u bAlternateSetting=%u\n", pUsbIns->iInstance, bInterfaceNumber, bAlternateSetting));
2056 Assert(bAlternateSetting == 0);
2057 return VINF_SUCCESS;
2058}
2059
2060
2061/**
2062 * @copydoc PDMUSBREG::pfnUsbSetConfiguration
2063 */
2064static DECLCALLBACK(int) usbMsdUsbSetConfiguration(PPDMUSBINS pUsbIns, uint8_t bConfigurationValue,
2065 const void *pvOldCfgDesc, const void *pvOldIfState, const void *pvNewCfgDesc)
2066{
2067 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2068 LogFlow(("usbMsdUsbSetConfiguration/#%u: bConfigurationValue=%u\n", pUsbIns->iInstance, bConfigurationValue));
2069 Assert(bConfigurationValue == 1);
2070 RTCritSectEnter(&pThis->CritSect);
2071
2072 /*
2073 * If the same config is applied more than once, it's a kind of reset.
2074 */
2075 if (pThis->bConfigurationValue == bConfigurationValue)
2076 usbMsdResetWorker(pThis, NULL, true /*fSetConfig*/); /** @todo figure out the exact difference */
2077 pThis->bConfigurationValue = bConfigurationValue;
2078
2079 RTCritSectLeave(&pThis->CritSect);
2080 return VINF_SUCCESS;
2081}
2082
2083
2084/**
2085 * @copydoc PDMUSBREG::pfnUsbGetDescriptorCache
2086 */
2087static DECLCALLBACK(PCPDMUSBDESCCACHE) usbMsdUsbGetDescriptorCache(PPDMUSBINS pUsbIns)
2088{
2089 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2090 LogFlow(("usbMsdUsbGetDescriptorCache/#%u:\n", pUsbIns->iInstance));
2091 if (pThis->pUsbIns->enmSpeed == VUSB_SPEED_SUPER)
2092 return pThis->fIsCdrom ? &g_UsbCdDescCacheSS : &g_UsbMsdDescCacheSS;
2093 else if (pThis->pUsbIns->enmSpeed == VUSB_SPEED_HIGH)
2094 return pThis->fIsCdrom ? &g_UsbCdDescCacheHS : &g_UsbMsdDescCacheHS;
2095 else
2096 return pThis->fIsCdrom ? &g_UsbCdDescCacheFS : &g_UsbMsdDescCacheFS;
2097}
2098
2099
2100/**
2101 * @copydoc PDMUSBREG::pfnUsbReset
2102 */
2103static DECLCALLBACK(int) usbMsdUsbReset(PPDMUSBINS pUsbIns, bool fResetOnLinux)
2104{
2105 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2106 LogFlow(("usbMsdUsbReset/#%u:\n", pUsbIns->iInstance));
2107 RTCritSectEnter(&pThis->CritSect);
2108
2109 int rc = usbMsdResetWorker(pThis, NULL, false /*fSetConfig*/);
2110
2111 RTCritSectLeave(&pThis->CritSect);
2112 return rc;
2113}
2114
2115
2116/**
2117 * @copydoc PDMUSBREG::pfnVMSuspend
2118 */
2119static DECLCALLBACK(void) usbMsdVMSuspend(PPDMUSBINS pUsbIns)
2120{
2121 LogFlow(("usbMsdVMSuspend/#%u:\n", pUsbIns->iInstance));
2122 usbMsdSuspendOrPowerOff(pUsbIns);
2123}
2124
2125
2126/**
2127 * @copydoc PDMUSBREG::pfnVMSuspend
2128 */
2129static DECLCALLBACK(void) usbMsdVMPowerOff(PPDMUSBINS pUsbIns)
2130{
2131 LogFlow(("usbMsdVMPowerOff/#%u:\n", pUsbIns->iInstance));
2132 usbMsdSuspendOrPowerOff(pUsbIns);
2133}
2134
2135
2136/**
2137 * @copydoc PDMUSBREG::pfnDriverAttach
2138 */
2139static DECLCALLBACK(int) usbMsdDriverAttach(PPDMUSBINS pUsbIns, unsigned iLUN, uint32_t fFlags)
2140{
2141 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2142 int rc;
2143
2144 LogFlow(("usbMsdDetach/#%u:\n", pUsbIns->iInstance));
2145
2146 AssertMsg(iLUN == 0, ("UsbMsd: No other LUN than 0 is supported\n"));
2147 AssertMsg(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
2148 ("UsbMsd: Device does not support hotplugging\n"));
2149
2150 /* the usual paranoia */
2151 AssertRelease(!pThis->Lun0.pIBase);
2152 AssertRelease(!pThis->Lun0.pIScsiConnector);
2153
2154 /*
2155 * Try attach the block device and get the interfaces,
2156 * required as well as optional.
2157 */
2158 rc = PDMUsbHlpDriverAttach(pUsbIns, iLUN, &pThis->Lun0.IBase, &pThis->Lun0.pIBase, NULL);
2159 if (RT_SUCCESS(rc))
2160 {
2161 /* Get SCSI connector interface. */
2162 pThis->Lun0.pIScsiConnector = PDMIBASE_QUERY_INTERFACE(pThis->Lun0.pIBase, PDMISCSICONNECTOR);
2163 AssertMsgReturn(pThis->Lun0.pIScsiConnector, ("Missing SCSI interface below\n"), VERR_PDM_MISSING_INTERFACE);
2164 }
2165 else
2166 AssertMsgFailed(("Failed to attach LUN#%d. rc=%Rrc\n", iLUN, rc));
2167
2168 if (RT_FAILURE(rc))
2169 {
2170 pThis->Lun0.pIBase = NULL;
2171 pThis->Lun0.pIScsiConnector = NULL;
2172 }
2173
2174 /*
2175 * Find out what kind of device we are.
2176 */
2177 PDMSCSILUNTYPE enmLunType;
2178 pThis->fIsCdrom = false;
2179 rc = pThis->Lun0.pIScsiConnector->pfnQueryLUNType(pThis->Lun0.pIScsiConnector, 0 /*iLun*/, &enmLunType);
2180 if (RT_SUCCESS(rc))
2181 {
2182 /* Anything else will be reported as a hard disk. */
2183 if (enmLunType == PDMSCSILUNTYPE_MMC)
2184 pThis->fIsCdrom = true;
2185 }
2186
2187 return rc;
2188}
2189
2190
2191/**
2192 * @copydoc PDMUSBREG::pfnDriverDetach
2193 */
2194static DECLCALLBACK(void) usbMsdDriverDetach(PPDMUSBINS pUsbIns, unsigned iLUN, uint32_t fFlags)
2195{
2196 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2197
2198 LogFlow(("usbMsdDetach/#%u:\n", pUsbIns->iInstance));
2199
2200 AssertMsg(iLUN == 0, ("UsbMsd: No other LUN than 0 is supported\n"));
2201 AssertMsg(fFlags & PDM_TACH_FLAGS_NOT_HOT_PLUG,
2202 ("UsbMsd: Device does not support hotplugging\n"));
2203
2204 /*
2205 * Zero some important members.
2206 */
2207 pThis->Lun0.pIBase = NULL;
2208 pThis->Lun0.pIScsiConnector = NULL;
2209}
2210
2211
2212/**
2213 * @callback_method_impl{FNPDMDEVASYNCNOTIFY,
2214 * Callback employed by usbMsdVMReset.}
2215 */
2216static DECLCALLBACK(bool) usbMsdIsAsyncResetDone(PPDMUSBINS pUsbIns)
2217{
2218 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2219
2220 if (!usbMsdAllAsyncIOIsFinished(pUsbIns))
2221 return false;
2222 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
2223
2224 int rc = usbMsdResetWorker(pThis, NULL, false /*fSetConfig*/);
2225 AssertRC(rc);
2226 return true;
2227}
2228
2229/**
2230 * @interface_method_impl{PDMDEVREG,pfnReset}
2231 */
2232static DECLCALLBACK(void) usbMsdVMReset(PPDMUSBINS pUsbIns)
2233{
2234 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2235
2236 ASMAtomicWriteBool(&pThis->fSignalIdle, true);
2237 if (!usbMsdAllAsyncIOIsFinished(pUsbIns))
2238 PDMUsbHlpSetAsyncNotification(pUsbIns, usbMsdIsAsyncResetDone);
2239 else
2240 {
2241 ASMAtomicWriteBool(&pThis->fSignalIdle, false);
2242 int rc = usbMsdResetWorker(pThis, NULL, false /*fSetConfig*/);
2243 AssertRC(rc);
2244 }
2245}
2246
2247
2248/**
2249 * @copydoc PDMUSBREG::pfnDestruct
2250 */
2251static void usbMsdDestruct(PPDMUSBINS pUsbIns)
2252{
2253 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2254 LogFlow(("usbMsdDestruct/#%u:\n", pUsbIns->iInstance));
2255
2256 if (RTCritSectIsInitialized(&pThis->CritSect))
2257 {
2258 RTCritSectEnter(&pThis->CritSect);
2259 RTCritSectLeave(&pThis->CritSect);
2260 RTCritSectDelete(&pThis->CritSect);
2261 }
2262
2263 if (pThis->pReq)
2264 {
2265 usbMsdReqFree(pThis->pReq);
2266 pThis->pReq = NULL;
2267 }
2268
2269 if (pThis->hEvtDoneQueue != NIL_RTSEMEVENT)
2270 {
2271 RTSemEventDestroy(pThis->hEvtDoneQueue);
2272 pThis->hEvtDoneQueue = NIL_RTSEMEVENT;
2273 }
2274
2275 if (pThis->hEvtReset != NIL_RTSEMEVENTMULTI)
2276 {
2277 RTSemEventMultiDestroy(pThis->hEvtReset);
2278 pThis->hEvtReset = NIL_RTSEMEVENTMULTI;
2279 }
2280}
2281
2282
2283/**
2284 * @copydoc PDMUSBREG::pfnConstruct
2285 */
2286static DECLCALLBACK(int) usbMsdConstruct(PPDMUSBINS pUsbIns, int iInstance, PCFGMNODE pCfg, PCFGMNODE pCfgGlobal)
2287{
2288 PUSBMSD pThis = PDMINS_2_DATA(pUsbIns, PUSBMSD);
2289 Log(("usbMsdConstruct/#%u:\n", iInstance));
2290
2291 /*
2292 * Perform the basic structure initialization first so the destructor
2293 * will not misbehave.
2294 */
2295 pThis->pUsbIns = pUsbIns;
2296 pThis->hEvtDoneQueue = NIL_RTSEMEVENT;
2297 pThis->hEvtReset = NIL_RTSEMEVENTMULTI;
2298 pThis->Lun0.IBase.pfnQueryInterface = usbMsdLun0QueryInterface;
2299 pThis->Lun0.IScsiPort.pfnSCSIRequestCompleted = usbMsdLun0ScsiRequestCompleted;
2300 pThis->Lun0.IScsiPort.pfnQueryDeviceLocation = usbMsdLun0QueryDeviceLocation;
2301 usbMsdQueueInit(&pThis->ToHostQueue);
2302 usbMsdQueueInit(&pThis->DoneQueue);
2303
2304 int rc = RTCritSectInit(&pThis->CritSect);
2305 AssertRCReturn(rc, rc);
2306
2307 rc = RTSemEventCreate(&pThis->hEvtDoneQueue);
2308 AssertRCReturn(rc, rc);
2309
2310 rc = RTSemEventMultiCreate(&pThis->hEvtReset);
2311 AssertRCReturn(rc, rc);
2312
2313 /*
2314 * Validate and read the configuration.
2315 */
2316 rc = CFGMR3ValidateConfig(pCfg, "/", "", "", "UsbMsd", iInstance);
2317 if (RT_FAILURE(rc))
2318 return rc;
2319
2320 /*
2321 * Attach the SCSI driver.
2322 */
2323 rc = PDMUsbHlpDriverAttach(pUsbIns, 0 /*iLun*/, &pThis->Lun0.IBase, &pThis->Lun0.pIBase, "SCSI Port");
2324 if (RT_FAILURE(rc))
2325 return PDMUsbHlpVMSetError(pUsbIns, rc, RT_SRC_POS, N_("MSD failed to attach SCSI driver"));
2326 pThis->Lun0.pIScsiConnector = PDMIBASE_QUERY_INTERFACE(pThis->Lun0.pIBase, PDMISCSICONNECTOR);
2327 if (!pThis->Lun0.pIScsiConnector)
2328 return PDMUsbHlpVMSetError(pUsbIns, VERR_PDM_MISSING_INTERFACE_BELOW, RT_SRC_POS,
2329 N_("MSD failed to query the PDMISCSICONNECTOR from the driver below it"));
2330
2331 /*
2332 * Find out what kind of device we are.
2333 */
2334 PDMSCSILUNTYPE enmLunType;
2335 pThis->fIsCdrom = false;
2336 rc = pThis->Lun0.pIScsiConnector->pfnQueryLUNType(pThis->Lun0.pIScsiConnector, 0 /*iLun*/, &enmLunType);
2337 if (RT_SUCCESS(rc))
2338 {
2339 /* Anything else will be reported as a hard disk. */
2340 if (enmLunType == PDMSCSILUNTYPE_MMC)
2341 pThis->fIsCdrom = true;
2342 }
2343
2344 /*
2345 * Register the saved state data unit.
2346 */
2347 rc = PDMUsbHlpSSMRegister(pUsbIns, USB_MSD_SAVED_STATE_VERSION, sizeof(*pThis),
2348 NULL, usbMsdLiveExec, NULL,
2349 usbMsdSavePrep, usbMsdSaveExec, NULL,
2350 usbMsdLoadPrep, usbMsdLoadExec, NULL);
2351 if (RT_FAILURE(rc))
2352 return PDMUsbHlpVMSetError(pUsbIns, rc, RT_SRC_POS,
2353 N_("MSD failed to register SSM save state handlers"));
2354
2355 return VINF_SUCCESS;
2356}
2357
2358
2359/**
2360 * The USB Mass Storage Device (MSD) registration record.
2361 */
2362const PDMUSBREG g_UsbMsd =
2363{
2364 /* u32Version */
2365 PDM_USBREG_VERSION,
2366 /* szName */
2367 "Msd",
2368 /* pszDescription */
2369 "USB Mass Storage Device, one LUN.",
2370 /* fFlags */
2371 PDM_USBREG_HIGHSPEED_CAPABLE | PDM_USBREG_SUPERSPEED_CAPABLE
2372 | PDM_USBREG_SAVED_STATE_SUPPORTED,
2373 /* cMaxInstances */
2374 ~0U,
2375 /* cbInstance */
2376 sizeof(USBMSD),
2377 /* pfnConstruct */
2378 usbMsdConstruct,
2379 /* pfnDestruct */
2380 usbMsdDestruct,
2381 /* pfnVMInitComplete */
2382 NULL,
2383 /* pfnVMPowerOn */
2384 NULL,
2385 /* pfnVMReset */
2386 usbMsdVMReset,
2387 /* pfnVMSuspend */
2388 usbMsdVMSuspend,
2389 /* pfnVMResume */
2390 NULL,
2391 /* pfnVMPowerOff */
2392 usbMsdVMPowerOff,
2393 /* pfnHotPlugged */
2394 NULL,
2395 /* pfnHotUnplugged */
2396 NULL,
2397 /* pfnDriverAttach */
2398 usbMsdDriverAttach,
2399 /* pfnDriverDetach */
2400 usbMsdDriverDetach,
2401 /* pfnQueryInterface */
2402 NULL,
2403 /* pfnUsbReset */
2404 usbMsdUsbReset,
2405 /* pfnUsbGetCachedDescriptors */
2406 usbMsdUsbGetDescriptorCache,
2407 /* pfnUsbSetConfiguration */
2408 usbMsdUsbSetConfiguration,
2409 /* pfnUsbSetInterface */
2410 usbMsdUsbSetInterface,
2411 /* pfnUsbClearHaltedEndpoint */
2412 usbMsdUsbClearHaltedEndpoint,
2413 /* pfnUrbNew */
2414 NULL/*usbMsdUrbNew*/,
2415 /* pfnQueue */
2416 usbMsdQueue,
2417 /* pfnUrbCancel */
2418 usbMsdUrbCancel,
2419 /* pfnUrbReap */
2420 usbMsdUrbReap,
2421 /* pfnWakeup */
2422 usbMsdWakeup,
2423 /* u32TheEnd */
2424 PDM_USBREG_VERSION
2425};
2426
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