VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvSCSI.cpp@ 49890

Last change on this file since 49890 was 48947, checked in by vboxsync, 11 years ago

Devices: Whitespace and svn:keyword cleanups by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 38.3 KB
Line 
1/* $Id: DrvSCSI.cpp 48947 2013-10-07 21:41:00Z vboxsync $ */
2/** @file
3 * VBox storage drivers: Generic SCSI command parser and execution driver
4 */
5
6/*
7 * Copyright (C) 2006-2012 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 DEBUG
22#define LOG_GROUP LOG_GROUP_DRV_SCSI
23#include <VBox/vmm/pdmdrv.h>
24#include <VBox/vmm/pdmifs.h>
25#include <VBox/vmm/pdmthread.h>
26#include <VBox/vscsi.h>
27#include <VBox/scsi.h>
28#include <iprt/asm.h>
29#include <iprt/assert.h>
30#include <iprt/mem.h>
31#include <iprt/req.h>
32#include <iprt/semaphore.h>
33#include <iprt/string.h>
34#include <iprt/uuid.h>
35
36#include "VBoxDD.h"
37
38/** The maximum number of release log entries per device. */
39#define MAX_LOG_REL_ERRORS 1024
40
41/**
42 * SCSI driver instance data.
43 *
44 * @implements PDMISCSICONNECTOR
45 * @implements PDMIBLOCKASYNCPORT
46 * @implements PDMIMOUNTNOTIFY
47 */
48typedef struct DRVSCSI
49{
50 /** Pointer driver instance. */
51 PPDMDRVINS pDrvIns;
52
53 /** Pointer to the attached driver's base interface. */
54 PPDMIBASE pDrvBase;
55 /** Pointer to the attached driver's block interface. */
56 PPDMIBLOCK pDrvBlock;
57 /** Pointer to the attached driver's async block interface. */
58 PPDMIBLOCKASYNC pDrvBlockAsync;
59 /** Pointer to the attached driver's block bios interface. */
60 PPDMIBLOCKBIOS pDrvBlockBios;
61 /** Pointer to the attached driver's mount interface. */
62 PPDMIMOUNT pDrvMount;
63 /** Pointer to the SCSI port interface of the device above. */
64 PPDMISCSIPORT pDevScsiPort;
65 /** pointer to the Led port interface of the dveice above. */
66 PPDMILEDPORTS pLedPort;
67 /** The scsi connector interface .*/
68 PDMISCSICONNECTOR ISCSIConnector;
69 /** The block port interface. */
70 PDMIBLOCKPORT IPort;
71 /** The optional block async port interface. */
72 PDMIBLOCKASYNCPORT IPortAsync;
73 /** The mount notify interface. */
74 PDMIMOUNTNOTIFY IMountNotify;
75 /** Fallback status LED state for this drive.
76 * This is used in case the device doesn't has a LED interface. */
77 PDMLED Led;
78 /** Pointer to the status LED for this drive. */
79 PPDMLED pLed;
80
81 /** VSCSI device handle. */
82 VSCSIDEVICE hVScsiDevice;
83 /** VSCSI LUN handle. */
84 VSCSILUN hVScsiLun;
85 /** I/O callbacks. */
86 VSCSILUNIOCALLBACKS VScsiIoCallbacks;
87
88 /** The dedicated I/O thread for the non async approach. */
89 PPDMTHREAD pAsyncIOThread;
90 /** Queue for passing the requests to the thread. */
91 RTREQQUEUE hQueueRequests;
92 /** Request that we've left pending on wakeup or reset. */
93 PRTREQ pPendingDummyReq;
94 /** Indicates whether PDMDrvHlpAsyncNotificationCompleted should be called by
95 * any of the dummy functions. */
96 bool volatile fDummySignal;
97 /** Release statistics: number of bytes written. */
98 STAMCOUNTER StatBytesWritten;
99 /** Release statistics: number of bytes read. */
100 STAMCOUNTER StatBytesRead;
101 /** Release statistics: Current I/O depth. */
102 volatile uint32_t StatIoDepth;
103 /** Errors printed in the release log. */
104 unsigned cErrors;
105 /** Mark the drive as having a non-rotational medium (i.e. as a SSD). */
106 bool fNonRotational;
107 /** Medium is readonly */
108 bool fReadonly;
109} DRVSCSI, *PDRVSCSI;
110
111/** Converts a pointer to DRVSCSI::ISCSIConnector to a PDRVSCSI. */
112#define PDMISCSICONNECTOR_2_DRVSCSI(pInterface) ( (PDRVSCSI)((uintptr_t)pInterface - RT_OFFSETOF(DRVSCSI, ISCSIConnector)) )
113/** Converts a pointer to DRVSCSI::IPortAsync to a PDRVSCSI. */
114#define PDMIBLOCKASYNCPORT_2_DRVSCSI(pInterface) ( (PDRVSCSI)((uintptr_t)pInterface - RT_OFFSETOF(DRVSCSI, IPortAsync)) )
115/** Converts a pointer to DRVSCSI::IMountNotify to PDRVSCSI. */
116#define PDMIMOUNTNOTIFY_2_DRVSCSI(pInterface) ( (PDRVSCSI)((uintptr_t)pInterface - RT_OFFSETOF(DRVSCSI, IMountNotify)) )
117/** Converts a pointer to DRVSCSI::IPort to a PDRVSCSI. */
118#define PDMIBLOCKPORT_2_DRVSCSI(pInterface) ( (PDRVSCSI)((uintptr_t)pInterface - RT_OFFSETOF(DRVSCSI, IPort)) )
119
120static bool drvscsiIsRedoPossible(int rc)
121{
122 if ( rc == VERR_DISK_FULL
123 || rc == VERR_FILE_TOO_BIG
124 || rc == VERR_BROKEN_PIPE
125 || rc == VERR_NET_CONNECTION_REFUSED)
126 return true;
127
128 return false;
129}
130
131static int drvscsiProcessRequestOne(PDRVSCSI pThis, VSCSIIOREQ hVScsiIoReq)
132{
133 int rc = VINF_SUCCESS;
134 VSCSIIOREQTXDIR enmTxDir;
135
136 enmTxDir = VSCSIIoReqTxDirGet(hVScsiIoReq);
137
138 switch (enmTxDir)
139 {
140 case VSCSIIOREQTXDIR_FLUSH:
141 {
142 rc = pThis->pDrvBlock->pfnFlush(pThis->pDrvBlock);
143 if ( RT_FAILURE(rc)
144 && pThis->cErrors++ < MAX_LOG_REL_ERRORS)
145 LogRel(("SCSI#%u: Flush returned rc=%Rrc\n",
146 pThis->pDrvIns->iInstance, rc));
147 break;
148 }
149 case VSCSIIOREQTXDIR_READ:
150 case VSCSIIOREQTXDIR_WRITE:
151 {
152 uint64_t uOffset = 0;
153 size_t cbTransfer = 0;
154 size_t cbSeg = 0;
155 PCRTSGSEG paSeg = NULL;
156 unsigned cSeg = 0;
157
158 rc = VSCSIIoReqParamsGet(hVScsiIoReq, &uOffset, &cbTransfer, &cSeg, &cbSeg,
159 &paSeg);
160 AssertRC(rc);
161
162 while (cbTransfer && cSeg)
163 {
164 size_t cbProcess = (cbTransfer < paSeg->cbSeg) ? cbTransfer : paSeg->cbSeg;
165
166 Log(("%s: uOffset=%llu cbProcess=%u\n", __FUNCTION__, uOffset, cbProcess));
167
168 if (enmTxDir == VSCSIIOREQTXDIR_READ)
169 {
170 pThis->pLed->Asserted.s.fReading = pThis->pLed->Actual.s.fReading = 1;
171 rc = pThis->pDrvBlock->pfnRead(pThis->pDrvBlock, uOffset,
172 paSeg->pvSeg, cbProcess);
173 pThis->pLed->Actual.s.fReading = 0;
174 if (RT_FAILURE(rc))
175 break;
176 STAM_REL_COUNTER_ADD(&pThis->StatBytesRead, cbProcess);
177 }
178 else
179 {
180 pThis->pLed->Asserted.s.fWriting = pThis->pLed->Actual.s.fWriting = 1;
181 rc = pThis->pDrvBlock->pfnWrite(pThis->pDrvBlock, uOffset,
182 paSeg->pvSeg, cbProcess);
183 pThis->pLed->Actual.s.fWriting = 0;
184 if (RT_FAILURE(rc))
185 break;
186 STAM_REL_COUNTER_ADD(&pThis->StatBytesWritten, cbProcess);
187 }
188
189 /* Go to the next entry. */
190 uOffset += cbProcess;
191 cbTransfer -= cbProcess;
192 paSeg++;
193 cSeg--;
194 }
195
196 if ( RT_FAILURE(rc)
197 && pThis->cErrors++ < MAX_LOG_REL_ERRORS)
198 LogRel(("SCSI#%u: %s at offset %llu (%u bytes left) returned rc=%Rrc\n",
199 pThis->pDrvIns->iInstance,
200 enmTxDir == VSCSIIOREQTXDIR_READ
201 ? "Read"
202 : "Write",
203 uOffset,
204 cbTransfer, rc));
205
206 break;
207 }
208 case VSCSIIOREQTXDIR_UNMAP:
209 {
210 PCRTRANGE paRanges;
211 unsigned cRanges;
212
213 rc = VSCSIIoReqUnmapParamsGet(hVScsiIoReq, &paRanges, &cRanges);
214 AssertRC(rc);
215
216 pThis->pLed->Asserted.s.fWriting = pThis->pLed->Actual.s.fWriting = 1;
217 rc = pThis->pDrvBlock->pfnDiscard(pThis->pDrvBlock, paRanges, cRanges);
218 pThis->pLed->Actual.s.fWriting = 0;
219
220 if ( RT_FAILURE(rc)
221 && pThis->cErrors++ < MAX_LOG_REL_ERRORS)
222 LogRel(("SCSI#%u: Unmap returned rc=%Rrc\n",
223 pThis->pDrvIns->iInstance, rc));
224
225 break;
226 }
227 default:
228 AssertMsgFailed(("Invalid transfer direction %d\n", enmTxDir));
229 }
230
231 if (RT_SUCCESS(rc))
232 VSCSIIoReqCompleted(hVScsiIoReq, rc, false /* fRedoPossible */);
233 else
234 VSCSIIoReqCompleted(hVScsiIoReq, rc, drvscsiIsRedoPossible(rc));
235
236 return VINF_SUCCESS;
237}
238
239static DECLCALLBACK(int) drvscsiGetSize(VSCSILUN hVScsiLun, void *pvScsiLunUser, uint64_t *pcbSize)
240{
241 PDRVSCSI pThis = (PDRVSCSI)pvScsiLunUser;
242
243 *pcbSize = pThis->pDrvBlock->pfnGetSize(pThis->pDrvBlock);
244
245 return VINF_SUCCESS;
246}
247
248
249static DECLCALLBACK(int) drvscsiGetSectorSize(VSCSILUN hVScsiLun, void *pvScsiLunUser, uint32_t *pcbSectorSize)
250{
251 PDRVSCSI pThis = (PDRVSCSI)pvScsiLunUser;
252
253 *pcbSectorSize = pThis->pDrvBlock->pfnGetSectorSize(pThis->pDrvBlock);
254
255 return VINF_SUCCESS;
256}
257static DECLCALLBACK(int) drvscsiSetLock(VSCSILUN hVScsiLun, void *pvScsiLunUser, bool fLocked)
258{
259 PDRVSCSI pThis = (PDRVSCSI)pvScsiLunUser;
260
261 if (fLocked)
262 pThis->pDrvMount->pfnLock(pThis->pDrvMount);
263 else
264 pThis->pDrvMount->pfnUnlock(pThis->pDrvMount);
265
266 return VINF_SUCCESS;
267}
268
269static int drvscsiTransferCompleteNotify(PPDMIBLOCKASYNCPORT pInterface, void *pvUser, int rc)
270{
271 PDRVSCSI pThis = PDMIBLOCKASYNCPORT_2_DRVSCSI(pInterface);
272 VSCSIIOREQ hVScsiIoReq = (VSCSIIOREQ)pvUser;
273 VSCSIIOREQTXDIR enmTxDir = VSCSIIoReqTxDirGet(hVScsiIoReq);
274
275 LogFlowFunc(("Request hVScsiIoReq=%#p completed\n", hVScsiIoReq));
276
277 if (enmTxDir == VSCSIIOREQTXDIR_READ)
278 pThis->pLed->Actual.s.fReading = 0;
279 else if ( enmTxDir == VSCSIIOREQTXDIR_WRITE
280 || enmTxDir == VSCSIIOREQTXDIR_UNMAP)
281 pThis->pLed->Actual.s.fWriting = 0;
282 else
283 AssertMsg(enmTxDir == VSCSIIOREQTXDIR_FLUSH, ("Invalid transfer direction %u\n", enmTxDir));
284
285 if (RT_SUCCESS(rc))
286 VSCSIIoReqCompleted(hVScsiIoReq, rc, false /* fRedoPossible */);
287 else
288 {
289 pThis->cErrors++;
290 if (pThis->cErrors < MAX_LOG_REL_ERRORS)
291 {
292 if (enmTxDir == VSCSIIOREQTXDIR_FLUSH)
293 LogRel(("SCSI#%u: Flush returned rc=%Rrc\n",
294 pThis->pDrvIns->iInstance, rc));
295 else if (enmTxDir == VSCSIIOREQTXDIR_UNMAP)
296 LogRel(("SCSI#%u: Unmap returned rc=%Rrc\n",
297 pThis->pDrvIns->iInstance, rc));
298 else
299 {
300 uint64_t uOffset = 0;
301 size_t cbTransfer = 0;
302 size_t cbSeg = 0;
303 PCRTSGSEG paSeg = NULL;
304 unsigned cSeg = 0;
305
306 VSCSIIoReqParamsGet(hVScsiIoReq, &uOffset, &cbTransfer,
307 &cSeg, &cbSeg, &paSeg);
308
309 LogRel(("SCSI#%u: %s at offset %llu (%u bytes left) returned rc=%Rrc\n",
310 pThis->pDrvIns->iInstance,
311 enmTxDir == VSCSIIOREQTXDIR_READ
312 ? "Read"
313 : "Write",
314 uOffset,
315 cbTransfer, rc));
316 }
317 }
318
319 VSCSIIoReqCompleted(hVScsiIoReq, rc, drvscsiIsRedoPossible(rc));
320 }
321
322 return VINF_SUCCESS;
323}
324
325static DECLCALLBACK(int) drvscsiReqTransferEnqueue(VSCSILUN hVScsiLun,
326 void *pvScsiLunUser,
327 VSCSIIOREQ hVScsiIoReq)
328{
329 int rc = VINF_SUCCESS;
330 PDRVSCSI pThis = (PDRVSCSI)pvScsiLunUser;
331
332 if (pThis->pDrvBlockAsync)
333 {
334 /* async I/O path. */
335 VSCSIIOREQTXDIR enmTxDir;
336
337 LogFlowFunc(("Enqueuing hVScsiIoReq=%#p\n", hVScsiIoReq));
338
339 enmTxDir = VSCSIIoReqTxDirGet(hVScsiIoReq);
340
341 switch (enmTxDir)
342 {
343 case VSCSIIOREQTXDIR_FLUSH:
344 {
345 rc = pThis->pDrvBlockAsync->pfnStartFlush(pThis->pDrvBlockAsync, hVScsiIoReq);
346 if ( RT_FAILURE(rc)
347 && rc != VERR_VD_ASYNC_IO_IN_PROGRESS
348 && pThis->cErrors++ < MAX_LOG_REL_ERRORS)
349 LogRel(("SCSI#%u: Flush returned rc=%Rrc\n",
350 pThis->pDrvIns->iInstance, rc));
351 break;
352 }
353 case VSCSIIOREQTXDIR_UNMAP:
354 {
355 PCRTRANGE paRanges;
356 unsigned cRanges;
357
358 rc = VSCSIIoReqUnmapParamsGet(hVScsiIoReq, &paRanges, &cRanges);
359 AssertRC(rc);
360
361 pThis->pLed->Asserted.s.fWriting = pThis->pLed->Actual.s.fWriting = 1;
362 rc = pThis->pDrvBlockAsync->pfnStartDiscard(pThis->pDrvBlockAsync, paRanges, cRanges, hVScsiIoReq);
363 if ( RT_FAILURE(rc)
364 && rc != VERR_VD_ASYNC_IO_IN_PROGRESS
365 && pThis->cErrors++ < MAX_LOG_REL_ERRORS)
366 LogRel(("SCSI#%u: Discard returned rc=%Rrc\n",
367 pThis->pDrvIns->iInstance, rc));
368 break;
369 }
370 case VSCSIIOREQTXDIR_READ:
371 case VSCSIIOREQTXDIR_WRITE:
372 {
373 uint64_t uOffset = 0;
374 size_t cbTransfer = 0;
375 size_t cbSeg = 0;
376 PCRTSGSEG paSeg = NULL;
377 unsigned cSeg = 0;
378
379 rc = VSCSIIoReqParamsGet(hVScsiIoReq, &uOffset, &cbTransfer,
380 &cSeg, &cbSeg, &paSeg);
381 AssertRC(rc);
382
383 if (enmTxDir == VSCSIIOREQTXDIR_READ)
384 {
385 pThis->pLed->Asserted.s.fReading = pThis->pLed->Actual.s.fReading = 1;
386 rc = pThis->pDrvBlockAsync->pfnStartRead(pThis->pDrvBlockAsync, uOffset,
387 paSeg, cSeg, cbTransfer,
388 hVScsiIoReq);
389 STAM_REL_COUNTER_ADD(&pThis->StatBytesRead, cbTransfer);
390 }
391 else
392 {
393 pThis->pLed->Asserted.s.fWriting = pThis->pLed->Actual.s.fWriting = 1;
394 rc = pThis->pDrvBlockAsync->pfnStartWrite(pThis->pDrvBlockAsync, uOffset,
395 paSeg, cSeg, cbTransfer,
396 hVScsiIoReq);
397 STAM_REL_COUNTER_ADD(&pThis->StatBytesWritten, cbTransfer);
398 }
399
400 if ( RT_FAILURE(rc)
401 && rc != VERR_VD_ASYNC_IO_IN_PROGRESS
402 && pThis->cErrors++ < MAX_LOG_REL_ERRORS)
403 LogRel(("SCSI#%u: %s at offset %llu (%u bytes left) returned rc=%Rrc\n",
404 pThis->pDrvIns->iInstance,
405 enmTxDir == VSCSIIOREQTXDIR_READ
406 ? "Read"
407 : "Write",
408 uOffset,
409 cbTransfer, rc));
410 break;
411 }
412 default:
413 AssertMsgFailed(("Invalid transfer direction %u\n", enmTxDir));
414 }
415
416 if (rc == VINF_VD_ASYNC_IO_FINISHED)
417 {
418 if (enmTxDir == VSCSIIOREQTXDIR_READ)
419 pThis->pLed->Actual.s.fReading = 0;
420 else if (enmTxDir == VSCSIIOREQTXDIR_WRITE)
421 pThis->pLed->Actual.s.fWriting = 0;
422 else
423 AssertMsg(enmTxDir == VSCSIIOREQTXDIR_FLUSH, ("Invalid transfer direction %u\n", enmTxDir));
424
425 VSCSIIoReqCompleted(hVScsiIoReq, VINF_SUCCESS, false);
426 rc = VINF_SUCCESS;
427 }
428 else if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
429 rc = VINF_SUCCESS;
430 else if (RT_FAILURE(rc))
431 {
432 if (enmTxDir == VSCSIIOREQTXDIR_READ)
433 pThis->pLed->Actual.s.fReading = 0;
434 else if (enmTxDir == VSCSIIOREQTXDIR_WRITE)
435 pThis->pLed->Actual.s.fWriting = 0;
436 else
437 AssertMsg(enmTxDir == VSCSIIOREQTXDIR_FLUSH, ("Invalid transfer direction %u\n", enmTxDir));
438
439 VSCSIIoReqCompleted(hVScsiIoReq, rc, drvscsiIsRedoPossible(rc));
440 rc = VINF_SUCCESS;
441 }
442 else
443 AssertMsgFailed(("Invalid return code rc=%Rrc\n", rc));
444 }
445 else
446 {
447 /* I/O thread. */
448 rc = RTReqQueueCallEx(pThis->hQueueRequests, NULL, 0, RTREQFLAGS_NO_WAIT,
449 (PFNRT)drvscsiProcessRequestOne, 2, pThis, hVScsiIoReq);
450 }
451
452 return rc;
453}
454
455static DECLCALLBACK(int) drvscsiGetFeatureFlags(VSCSILUN hVScsiLun,
456 void *pvScsiLunUser,
457 uint64_t *pfFeatures)
458{
459 int rc = VINF_SUCCESS;
460 PDRVSCSI pThis = (PDRVSCSI)pvScsiLunUser;
461
462 *pfFeatures = 0;
463
464 if ( pThis->pDrvBlock->pfnDiscard
465 || ( pThis->pDrvBlockAsync
466 && pThis->pDrvBlockAsync->pfnStartDiscard))
467 *pfFeatures |= VSCSI_LUN_FEATURE_UNMAP;
468
469 if (pThis->fNonRotational)
470 *pfFeatures |= VSCSI_LUN_FEATURE_NON_ROTATIONAL;
471
472 if (pThis->fReadonly)
473 *pfFeatures |= VSCSI_LUN_FEATURE_READONLY;
474
475 return VINF_SUCCESS;
476}
477
478static void drvscsiVScsiReqCompleted(VSCSIDEVICE hVScsiDevice, void *pVScsiDeviceUser,
479 void *pVScsiReqUser, int rcScsiCode, bool fRedoPossible,
480 int rcReq)
481{
482 PDRVSCSI pThis = (PDRVSCSI)pVScsiDeviceUser;
483
484 ASMAtomicDecU32(&pThis->StatIoDepth);
485
486 pThis->pDevScsiPort->pfnSCSIRequestCompleted(pThis->pDevScsiPort, (PPDMSCSIREQUEST)pVScsiReqUser,
487 rcScsiCode, fRedoPossible, rcReq);
488
489 if (RT_UNLIKELY(pThis->fDummySignal) && !pThis->StatIoDepth)
490 PDMDrvHlpAsyncNotificationCompleted(pThis->pDrvIns);
491}
492
493/**
494 * Dummy request function used by drvscsiReset to wait for all pending requests
495 * to complete prior to the device reset.
496 *
497 * @param pThis Pointer to the instance data.
498 * @returns VINF_SUCCESS.
499 */
500static int drvscsiAsyncIOLoopSyncCallback(PDRVSCSI pThis)
501{
502 if (pThis->fDummySignal)
503 PDMDrvHlpAsyncNotificationCompleted(pThis->pDrvIns);
504 return VINF_SUCCESS;
505}
506
507/**
508 * Request function to wakeup the thread.
509 *
510 * @param pThis Pointer to the instance data.
511 * @returns VWRN_STATE_CHANGED.
512 */
513static int drvscsiAsyncIOLoopWakeupFunc(PDRVSCSI pThis)
514{
515 if (pThis->fDummySignal)
516 PDMDrvHlpAsyncNotificationCompleted(pThis->pDrvIns);
517 return VWRN_STATE_CHANGED;
518}
519
520/**
521 * The thread function which processes the requests asynchronously.
522 *
523 * @returns VBox status code.
524 * @param pDrvIns Pointer to the driver instance data.
525 * @param pThread Pointer to the thread instance data.
526 */
527static int drvscsiAsyncIOLoop(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
528{
529 int rc = VINF_SUCCESS;
530 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
531
532 LogFlowFunc(("Entering async IO loop.\n"));
533
534 if (pThread->enmState == PDMTHREADSTATE_INITIALIZING)
535 return VINF_SUCCESS;
536
537 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
538 {
539 rc = RTReqQueueProcess(pThis->hQueueRequests, RT_INDEFINITE_WAIT);
540 AssertMsg(rc == VWRN_STATE_CHANGED, ("Left RTReqProcess and error code is not VWRN_STATE_CHANGED rc=%Rrc\n", rc));
541 }
542
543 return VINF_SUCCESS;
544}
545
546/**
547 * Deals with any pending dummy request
548 *
549 * @returns true if no pending dummy request, false if still pending.
550 * @param pThis The instance data.
551 * @param cMillies The number of milliseconds to wait for any
552 * pending request to finish.
553 */
554static bool drvscsiAsyncIOLoopNoPendingDummy(PDRVSCSI pThis, uint32_t cMillies)
555{
556 if (!pThis->pPendingDummyReq)
557 return true;
558 int rc = RTReqWait(pThis->pPendingDummyReq, cMillies);
559 if (RT_FAILURE(rc))
560 return false;
561 RTReqRelease(pThis->pPendingDummyReq);
562 pThis->pPendingDummyReq = NULL;
563 return true;
564}
565
566static int drvscsiAsyncIOLoopWakeup(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
567{
568 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
569 PRTREQ pReq;
570 int rc;
571
572 AssertMsgReturn(pThis->hQueueRequests != NIL_RTREQQUEUE, ("hQueueRequests is NULL\n"), VERR_INVALID_STATE);
573
574 if (!drvscsiAsyncIOLoopNoPendingDummy(pThis, 10000 /* 10 sec */))
575 {
576 LogRel(("drvscsiAsyncIOLoopWakeup#%u: previous dummy request is still pending\n", pDrvIns->iInstance));
577 return VERR_TIMEOUT;
578 }
579
580 rc = RTReqQueueCall(pThis->hQueueRequests, &pReq, 10000 /* 10 sec. */, (PFNRT)drvscsiAsyncIOLoopWakeupFunc, 1, pThis);
581 if (RT_SUCCESS(rc))
582 RTReqRelease(pReq);
583 else
584 {
585 pThis->pPendingDummyReq = pReq;
586 LogRel(("drvscsiAsyncIOLoopWakeup#%u: %Rrc pReq=%p\n", pDrvIns->iInstance, rc, pReq));
587 }
588
589 return rc;
590}
591
592/* -=-=-=-=- ISCSIConnector -=-=-=-=- */
593
594#ifdef DEBUG
595/**
596 * Dumps a SCSI request structure for debugging purposes.
597 *
598 * @returns nothing.
599 * @param pRequest Pointer to the request to dump.
600 */
601static void drvscsiDumpScsiRequest(PPDMSCSIREQUEST pRequest)
602{
603 Log(("Dump for pRequest=%#p Command: %s\n", pRequest, SCSICmdText(pRequest->pbCDB[0])));
604 Log(("cbCDB=%u\n", pRequest->cbCDB));
605 for (uint32_t i = 0; i < pRequest->cbCDB; i++)
606 Log(("pbCDB[%u]=%#x\n", i, pRequest->pbCDB[i]));
607 Log(("cbScatterGather=%u\n", pRequest->cbScatterGather));
608 Log(("cScatterGatherEntries=%u\n", pRequest->cScatterGatherEntries));
609 /* Print all scatter gather entries. */
610 for (uint32_t i = 0; i < pRequest->cScatterGatherEntries; i++)
611 {
612 Log(("ScatterGatherEntry[%u].cbSeg=%u\n", i, pRequest->paScatterGatherHead[i].cbSeg));
613 Log(("ScatterGatherEntry[%u].pvSeg=%#p\n", i, pRequest->paScatterGatherHead[i].pvSeg));
614 }
615 Log(("pvUser=%#p\n", pRequest->pvUser));
616}
617#endif
618
619/** @copydoc PDMISCSICONNECTOR::pfnSCSIRequestSend. */
620static DECLCALLBACK(int) drvscsiRequestSend(PPDMISCSICONNECTOR pInterface, PPDMSCSIREQUEST pSCSIRequest)
621{
622 int rc;
623 PDRVSCSI pThis = PDMISCSICONNECTOR_2_DRVSCSI(pInterface);
624 VSCSIREQ hVScsiReq;
625
626#ifdef DEBUG
627 drvscsiDumpScsiRequest(pSCSIRequest);
628#endif
629
630 rc = VSCSIDeviceReqCreate(pThis->hVScsiDevice, &hVScsiReq,
631 pSCSIRequest->uLogicalUnit,
632 pSCSIRequest->pbCDB,
633 pSCSIRequest->cbCDB,
634 pSCSIRequest->cbScatterGather,
635 pSCSIRequest->cScatterGatherEntries,
636 pSCSIRequest->paScatterGatherHead,
637 pSCSIRequest->pbSenseBuffer,
638 pSCSIRequest->cbSenseBuffer,
639 pSCSIRequest);
640 if (RT_FAILURE(rc))
641 return rc;
642
643 ASMAtomicIncU32(&pThis->StatIoDepth);
644 rc = VSCSIDeviceReqEnqueue(pThis->hVScsiDevice, hVScsiReq);
645
646 return rc;
647}
648
649/* -=-=-=-=- IBase -=-=-=-=- */
650
651/**
652 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
653 */
654static DECLCALLBACK(void *) drvscsiQueryInterface(PPDMIBASE pInterface, const char *pszIID)
655{
656 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
657 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
658
659 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMOUNT, pThis->pDrvMount);
660 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
661 PDMIBASE_RETURN_INTERFACE(pszIID, PDMISCSICONNECTOR, &pThis->ISCSIConnector);
662 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBLOCKPORT, &pThis->IPort);
663 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMOUNTNOTIFY, &pThis->IMountNotify);
664 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBLOCKASYNCPORT, &pThis->IPortAsync);
665 return NULL;
666}
667
668static DECLCALLBACK(int) drvscsiQueryDeviceLocation(PPDMIBLOCKPORT pInterface, const char **ppcszController,
669 uint32_t *piInstance, uint32_t *piLUN)
670{
671 PDRVSCSI pThis = PDMIBLOCKPORT_2_DRVSCSI(pInterface);
672
673 return pThis->pDevScsiPort->pfnQueryDeviceLocation(pThis->pDevScsiPort, ppcszController,
674 piInstance, piLUN);
675}
676
677/**
678 * Called when media is mounted.
679 *
680 * @param pInterface Pointer to the interface structure containing the called function pointer.
681 */
682static DECLCALLBACK(void) drvscsiMountNotify(PPDMIMOUNTNOTIFY pInterface)
683{
684 PDRVSCSI pThis = PDMIMOUNTNOTIFY_2_DRVSCSI(pInterface);
685 LogFlowFunc(("mounting LUN#%p\n", pThis->hVScsiLun));
686
687 /* Ignore the call if we're called while being attached. */
688 if (!pThis->pDrvBlock)
689 return;
690
691 /* Let the LUN know that a medium was mounted. */
692 VSCSILunMountNotify(pThis->hVScsiLun);
693}
694
695/**
696 * Called when media is unmounted
697 *
698 * @param pInterface Pointer to the interface structure containing the called function pointer.
699 */
700static DECLCALLBACK(void) drvscsiUnmountNotify(PPDMIMOUNTNOTIFY pInterface)
701{
702 PDRVSCSI pThis = PDMIMOUNTNOTIFY_2_DRVSCSI(pInterface);
703 LogFlowFunc(("unmounting LUN#%p\n", pThis->hVScsiLun));
704
705 /* Let the LUN know that the medium was unmounted. */
706 VSCSILunUnmountNotify(pThis->hVScsiLun);
707}
708
709/**
710 * Worker for drvscsiReset, drvscsiSuspend and drvscsiPowerOff.
711 *
712 * @param pDrvIns The driver instance.
713 * @param pfnAsyncNotify The async callback.
714 */
715static void drvscsiR3ResetOrSuspendOrPowerOff(PPDMDRVINS pDrvIns, PFNPDMDRVASYNCNOTIFY pfnAsyncNotify)
716{
717 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
718
719 if (!pThis->pDrvBlockAsync)
720 {
721 if (pThis->hQueueRequests != NIL_RTREQQUEUE)
722 return;
723
724 ASMAtomicWriteBool(&pThis->fDummySignal, true);
725 if (drvscsiAsyncIOLoopNoPendingDummy(pThis, 0 /*ms*/))
726 {
727 if (!RTReqQueueIsBusy(pThis->hQueueRequests))
728 {
729 ASMAtomicWriteBool(&pThis->fDummySignal, false);
730 return;
731 }
732
733 PRTREQ pReq;
734 int rc = RTReqQueueCall(pThis->hQueueRequests, &pReq, 0 /*ms*/, (PFNRT)drvscsiAsyncIOLoopSyncCallback, 1, pThis);
735 if (RT_SUCCESS(rc))
736 {
737 ASMAtomicWriteBool(&pThis->fDummySignal, false);
738 RTReqRelease(pReq);
739 return;
740 }
741
742 pThis->pPendingDummyReq = pReq;
743 }
744 }
745 else
746 {
747 if (pThis->StatIoDepth > 0)
748 {
749 ASMAtomicWriteBool(&pThis->fDummySignal, true);
750 }
751 return;
752 }
753
754 PDMDrvHlpSetAsyncNotification(pDrvIns, pfnAsyncNotify);
755}
756
757/**
758 * Callback employed by drvscsiSuspend and drvscsiPowerOff.
759 *
760 * @returns true if we've quiesced, false if we're still working.
761 * @param pDrvIns The driver instance.
762 */
763static DECLCALLBACK(bool) drvscsiIsAsyncSuspendOrPowerOffDone(PPDMDRVINS pDrvIns)
764{
765 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
766
767 if (pThis->pDrvBlockAsync)
768 {
769 if (pThis->StatIoDepth > 0)
770 return false;
771 else
772 return true;
773 }
774 else
775 {
776 if (!drvscsiAsyncIOLoopNoPendingDummy(pThis, 0 /*ms*/))
777 return false;
778 ASMAtomicWriteBool(&pThis->fDummySignal, false);
779 PDMR3ThreadSuspend(pThis->pAsyncIOThread);
780 return true;
781 }
782}
783
784/**
785 * @copydoc FNPDMDRVPOWEROFF
786 */
787static DECLCALLBACK(void) drvscsiPowerOff(PPDMDRVINS pDrvIns)
788{
789 drvscsiR3ResetOrSuspendOrPowerOff(pDrvIns, drvscsiIsAsyncSuspendOrPowerOffDone);
790}
791
792/**
793 * @copydoc FNPDMDRVSUSPEND
794 */
795static DECLCALLBACK(void) drvscsiSuspend(PPDMDRVINS pDrvIns)
796{
797 drvscsiR3ResetOrSuspendOrPowerOff(pDrvIns, drvscsiIsAsyncSuspendOrPowerOffDone);
798}
799
800/**
801 * Callback employed by drvscsiReset.
802 *
803 * @returns true if we've quiesced, false if we're still working.
804 * @param pDrvIns The driver instance.
805 */
806static DECLCALLBACK(bool) drvscsiIsAsyncResetDone(PPDMDRVINS pDrvIns)
807{
808 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
809
810 if (pThis->pDrvBlockAsync)
811 {
812 if (pThis->StatIoDepth > 0)
813 return false;
814 else
815 return true;
816 }
817 else
818 {
819 if (!drvscsiAsyncIOLoopNoPendingDummy(pThis, 0 /*ms*/))
820 return false;
821 ASMAtomicWriteBool(&pThis->fDummySignal, false);
822 return true;
823 }
824}
825
826/**
827 * @copydoc FNPDMDRVRESET
828 */
829static DECLCALLBACK(void) drvscsiReset(PPDMDRVINS pDrvIns)
830{
831 drvscsiR3ResetOrSuspendOrPowerOff(pDrvIns, drvscsiIsAsyncResetDone);
832}
833
834/**
835 * Destruct a driver instance.
836 *
837 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
838 * resources can be freed correctly.
839 *
840 * @param pDrvIns The driver instance data.
841 */
842static DECLCALLBACK(void) drvscsiDestruct(PPDMDRVINS pDrvIns)
843{
844 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
845 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
846
847 if (pThis->hQueueRequests != NIL_RTREQQUEUE)
848 {
849 if (!drvscsiAsyncIOLoopNoPendingDummy(pThis, 100 /*ms*/))
850 LogRel(("drvscsiDestruct#%u: previous dummy request is still pending\n", pDrvIns->iInstance));
851
852 int rc = RTReqQueueDestroy(pThis->hQueueRequests);
853 AssertMsgRC(rc, ("Failed to destroy queue rc=%Rrc\n", rc));
854 pThis->hQueueRequests = NIL_RTREQQUEUE;
855 }
856
857 /* Free the VSCSI device and LUN handle. */
858 if (pThis->hVScsiDevice)
859 {
860 VSCSILUN hVScsiLun;
861 int rc = VSCSIDeviceLunDetach(pThis->hVScsiDevice, 0, &hVScsiLun);
862 AssertRC(rc);
863
864 Assert(hVScsiLun == pThis->hVScsiLun);
865 rc = VSCSILunDestroy(hVScsiLun);
866 AssertRC(rc);
867 rc = VSCSIDeviceDestroy(pThis->hVScsiDevice);
868 AssertRC(rc);
869
870 pThis->hVScsiDevice = NULL;
871 pThis->hVScsiLun = NULL;
872 }
873}
874
875/**
876 * Construct a block driver instance.
877 *
878 * @copydoc FNPDMDRVCONSTRUCT
879 */
880static DECLCALLBACK(int) drvscsiConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
881{
882 int rc = VINF_SUCCESS;
883 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
884 LogFlowFunc(("pDrvIns=%#p pCfg=%#p\n", pDrvIns, pCfg));
885 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
886
887 /*
888 * Initialize the instance data.
889 */
890 pThis->pDrvIns = pDrvIns;
891 pThis->ISCSIConnector.pfnSCSIRequestSend = drvscsiRequestSend;
892
893 pDrvIns->IBase.pfnQueryInterface = drvscsiQueryInterface;
894
895 pThis->IMountNotify.pfnMountNotify = drvscsiMountNotify;
896 pThis->IMountNotify.pfnUnmountNotify = drvscsiUnmountNotify;
897 pThis->IPort.pfnQueryDeviceLocation = drvscsiQueryDeviceLocation;
898 pThis->IPortAsync.pfnTransferCompleteNotify = drvscsiTransferCompleteNotify;
899 pThis->hQueueRequests = NIL_RTREQQUEUE;
900
901 /* Query the SCSI port interface above. */
902 pThis->pDevScsiPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMISCSIPORT);
903 AssertMsgReturn(pThis->pDevScsiPort, ("Missing SCSI port interface above\n"), VERR_PDM_MISSING_INTERFACE);
904
905 /* Query the optional LED interface above. */
906 pThis->pLedPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMILEDPORTS);
907 if (pThis->pLedPort != NULL)
908 {
909 /* Get The Led. */
910 rc = pThis->pLedPort->pfnQueryStatusLed(pThis->pLedPort, 0, &pThis->pLed);
911 if (RT_FAILURE(rc))
912 pThis->pLed = &pThis->Led;
913 }
914 else
915 pThis->pLed = &pThis->Led;
916
917 /*
918 * Validate and read configuration.
919 */
920 if (!CFGMR3AreValuesValid(pCfg, "NonRotationalMedium\0Readonly\0"))
921 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
922 N_("SCSI configuration error: unknown option specified"));
923
924 rc = CFGMR3QueryBoolDef(pCfg, "NonRotationalMedium", &pThis->fNonRotational, false);
925 if (RT_FAILURE(rc))
926 return PDMDRV_SET_ERROR(pDrvIns, rc,
927 N_("SCSI configuration error: failed to read \"NonRotationalMedium\" as boolean"));
928
929 rc = CFGMR3QueryBoolDef(pCfg, "Readonly", &pThis->fReadonly, false);
930 if (RT_FAILURE(rc))
931 return PDMDRV_SET_ERROR(pDrvIns, rc,
932 N_("SCSI configuration error: failed to read \"Readonly\" as boolean"));
933
934 /*
935 * Try attach driver below and query it's block interface.
936 */
937 rc = PDMDrvHlpAttach(pDrvIns, fFlags, &pThis->pDrvBase);
938 AssertMsgReturn(RT_SUCCESS(rc), ("Attaching driver below failed rc=%Rrc\n", rc), rc);
939
940 /*
941 * Query the block and blockbios interfaces.
942 */
943 pThis->pDrvBlock = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBLOCK);
944 if (!pThis->pDrvBlock)
945 {
946 AssertMsgFailed(("Configuration error: No block interface!\n"));
947 return VERR_PDM_MISSING_INTERFACE;
948 }
949 pThis->pDrvBlockBios = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBLOCKBIOS);
950 if (!pThis->pDrvBlockBios)
951 {
952 AssertMsgFailed(("Configuration error: No block BIOS interface!\n"));
953 return VERR_PDM_MISSING_INTERFACE;
954 }
955
956 pThis->pDrvMount = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIMOUNT);
957
958 /* Try to get the optional async block interface. */
959 pThis->pDrvBlockAsync = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBLOCKASYNC);
960
961 PDMBLOCKTYPE enmType = pThis->pDrvBlock->pfnGetType(pThis->pDrvBlock);
962 VSCSILUNTYPE enmLunType;
963 switch (enmType)
964 {
965 case PDMBLOCKTYPE_HARD_DISK:
966 enmLunType = VSCSILUNTYPE_SBC;
967 break;
968 case PDMBLOCKTYPE_CDROM:
969 case PDMBLOCKTYPE_DVD:
970 enmLunType = VSCSILUNTYPE_MMC;
971 break;
972 default:
973 return PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_UNSUPPORTED_BLOCK_TYPE, RT_SRC_POS,
974 N_("Only hard disks and CD/DVD-ROMs are currently supported as SCSI devices (enmType=%d)"),
975 enmType);
976 }
977 if ( ( enmType == PDMBLOCKTYPE_DVD
978 || enmType == PDMBLOCKTYPE_CDROM)
979 && !pThis->pDrvMount)
980 {
981 AssertMsgFailed(("Internal error: cdrom without a mountable interface\n"));
982 return VERR_INTERNAL_ERROR;
983 }
984
985 /* Create VSCSI device and LUN. */
986 pThis->VScsiIoCallbacks.pfnVScsiLunMediumGetSize = drvscsiGetSize;
987 pThis->VScsiIoCallbacks.pfnVScsiLunMediumGetSectorSize = drvscsiGetSectorSize;
988 pThis->VScsiIoCallbacks.pfnVScsiLunReqTransferEnqueue = drvscsiReqTransferEnqueue;
989 pThis->VScsiIoCallbacks.pfnVScsiLunGetFeatureFlags = drvscsiGetFeatureFlags;
990 pThis->VScsiIoCallbacks.pfnVScsiLunMediumSetLock = drvscsiSetLock;
991
992 rc = VSCSIDeviceCreate(&pThis->hVScsiDevice, drvscsiVScsiReqCompleted, pThis);
993 AssertMsgReturn(RT_SUCCESS(rc), ("Failed to create VSCSI device rc=%Rrc\n"), rc);
994 rc = VSCSILunCreate(&pThis->hVScsiLun, enmLunType, &pThis->VScsiIoCallbacks,
995 pThis);
996 AssertMsgReturn(RT_SUCCESS(rc), ("Failed to create VSCSI LUN rc=%Rrc\n"), rc);
997 rc = VSCSIDeviceLunAttach(pThis->hVScsiDevice, pThis->hVScsiLun, 0);
998 AssertMsgReturn(RT_SUCCESS(rc), ("Failed to attached the LUN to the SCSI device\n"), rc);
999
1000 //@todo: This is a very hacky way of telling the LUN whether a medium was mounted.
1001 // The mount/unmount interface doesn't work in a very sensible manner!
1002 if (pThis->pDrvMount)
1003 {
1004 if (pThis->pDrvBlock->pfnGetSize(pThis->pDrvBlock))
1005 {
1006 rc = VINF_SUCCESS; VSCSILunMountNotify(pThis->hVScsiLun);
1007 AssertMsgReturn(RT_SUCCESS(rc), ("Failed to notify the LUN of media being mounted\n"), rc);
1008 }
1009 else
1010 {
1011 rc = VINF_SUCCESS; VSCSILunUnmountNotify(pThis->hVScsiLun);
1012 AssertMsgReturn(RT_SUCCESS(rc), ("Failed to notify the LUN of media being unmounted\n"), rc);
1013 }
1014 }
1015
1016 /* Register statistics counter. */
1017 /** @todo aeichner: Find a way to put the instance number of the attached
1018 * controller device when we support more than one controller of the same type.
1019 * At the moment we have the 0 hardcoded. */
1020 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatBytesRead, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
1021 "Amount of data read.", "/Devices/SCSI0/%d/ReadBytes", pDrvIns->iInstance);
1022 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatBytesWritten, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
1023 "Amount of data written.", "/Devices/SCSI0/%d/WrittenBytes", pDrvIns->iInstance);
1024
1025 pThis->StatIoDepth = 0;
1026
1027 PDMDrvHlpSTAMRegisterF(pDrvIns, (void *)&pThis->StatIoDepth, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT,
1028 "Number of active tasks.", "/Devices/SCSI0/%d/IoDepth", pDrvIns->iInstance);
1029
1030 if (!pThis->pDrvBlockAsync)
1031 {
1032 /* Create request queue. */
1033 rc = RTReqQueueCreate(&pThis->hQueueRequests);
1034 AssertMsgReturn(RT_SUCCESS(rc), ("Failed to create request queue rc=%Rrc\n"), rc);
1035 /* Create I/O thread. */
1036 rc = PDMDrvHlpThreadCreate(pDrvIns, &pThis->pAsyncIOThread, pThis, drvscsiAsyncIOLoop,
1037 drvscsiAsyncIOLoopWakeup, 0, RTTHREADTYPE_IO, "SCSI async IO");
1038 AssertMsgReturn(RT_SUCCESS(rc), ("Failed to create async I/O thread rc=%Rrc\n"), rc);
1039
1040 LogRel(("SCSI#%d: using normal I/O\n", pDrvIns->iInstance));
1041 }
1042 else
1043 LogRel(("SCSI#%d: using async I/O\n", pDrvIns->iInstance));
1044
1045 if ( pThis->pDrvBlock->pfnDiscard
1046 || ( pThis->pDrvBlockAsync
1047 && pThis->pDrvBlockAsync->pfnStartDiscard))
1048 LogRel(("SCSI#%d: Enabled UNMAP support\n"));
1049
1050 return VINF_SUCCESS;
1051}
1052
1053/**
1054 * SCSI driver registration record.
1055 */
1056const PDMDRVREG g_DrvSCSI =
1057{
1058 /* u32Version */
1059 PDM_DRVREG_VERSION,
1060 /* szName */
1061 "SCSI",
1062 /* szRCMod */
1063 "",
1064 /* szR0Mod */
1065 "",
1066 /* pszDescription */
1067 "Generic SCSI driver.",
1068 /* fFlags */
1069 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1070 /* fClass. */
1071 PDM_DRVREG_CLASS_SCSI,
1072 /* cMaxInstances */
1073 ~0U,
1074 /* cbInstance */
1075 sizeof(DRVSCSI),
1076 /* pfnConstruct */
1077 drvscsiConstruct,
1078 /* pfnDestruct */
1079 drvscsiDestruct,
1080 /* pfnRelocate */
1081 NULL,
1082 /* pfnIOCtl */
1083 NULL,
1084 /* pfnPowerOn */
1085 NULL,
1086 /* pfnReset */
1087 drvscsiReset,
1088 /* pfnSuspend */
1089 drvscsiSuspend,
1090 /* pfnResume */
1091 NULL,
1092 /* pfnAttach */
1093 NULL,
1094 /* pfnDetach */
1095 NULL,
1096 /* pfnPowerOff */
1097 drvscsiPowerOff,
1098 /* pfnSoftReset */
1099 NULL,
1100 /* u32EndVersion */
1101 PDM_DRVREG_VERSION
1102};
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