VirtualBox

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

Last change on this file since 27340 was 26837, checked in by vboxsync, 15 years ago

SCSI: Do not assert if there is no sense buffer

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 38.6 KB
Line 
1/* $Id: DrvSCSI.cpp 26837 2010-02-26 12:35:54Z vboxsync $ */
2/** @file
3 * VBox storage drivers: Generic SCSI command parser and execution driver
4 */
5
6/*
7 * Copyright (C) 2006-2010 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22/*******************************************************************************
23* Header Files *
24*******************************************************************************/
25//#define DEBUG
26#define LOG_GROUP LOG_GROUP_DRV_SCSI
27#include <VBox/pdmdrv.h>
28#include <VBox/pdmifs.h>
29#include <VBox/pdmthread.h>
30#include <VBox/scsi.h>
31#include <iprt/assert.h>
32#include <iprt/mem.h>
33#include <iprt/req.h>
34#include <iprt/semaphore.h>
35#include <iprt/string.h>
36#include <iprt/uuid.h>
37
38#include "Builtins.h"
39
40/**
41 * SCSI driver instance data.
42 *
43 * @implements PDMISCSICONNECTOR
44 * @implements PDMIBLOCKASYNCPORT
45 * @implements PDMIMOUNTNOTIFY
46 */
47typedef struct DRVSCSI
48{
49 /** Pointer driver instance. */
50 PPDMDRVINS pDrvIns;
51
52 /** Pointer to the attached driver's base interface. */
53 PPDMIBASE pDrvBase;
54 /** Pointer to the attached driver's block interface. */
55 PPDMIBLOCK pDrvBlock;
56 /** Pointer to the attached driver's async block interface. */
57 PPDMIBLOCKASYNC pDrvBlockAsync;
58 /** Pointer to the attached driver's block bios interface. */
59 PPDMIBLOCKBIOS pDrvBlockBios;
60 /** Pointer to the attached driver's mount interface. */
61 PPDMIMOUNT pDrvMount;
62 /** Pointer to the SCSI port interface of the device above. */
63 PPDMISCSIPORT pDevScsiPort;
64 /** pointer to the Led port interface of the dveice above. */
65 PPDMILEDPORTS pLedPort;
66 /** The scsi connector interface .*/
67 PDMISCSICONNECTOR ISCSIConnector;
68 /** The block port interface. */
69 PDMIBLOCKPORT IPort;
70#if 0 /* these interfaces aren't implemented */
71 /** The optional block async port interface. */
72 PDMIBLOCKASYNCPORT IPortAsync;
73 /** The mount notify interface. */
74 PDMIMOUNTNOTIFY IMountNotify;
75#endif
76 /** Fallback status LED state for this drive.
77 * This is used in case the device doesn't has a LED interface. */
78 PDMLED Led;
79 /** Pointer to the status LED for this drive. */
80 PPDMLED pLed;
81
82 /** Device type. */
83 PDMBLOCKTYPE enmType;
84 /** BIOS PCHS Geometry. */
85 PDMMEDIAGEOMETRY PCHSGeometry;
86 /** BIOS LCHS Geometry. */
87 PDMMEDIAGEOMETRY LCHSGeometry;
88 /** Number of sectors this device has. */
89 uint64_t cSectors;
90
91 /** The dedicated I/O thread for the non async approach. */
92 PPDMTHREAD pAsyncIOThread;
93 /** Queue for passing the requests to the thread. */
94 PRTREQQUEUE pQueueRequests;
95 /** Request that we've left pending on wakeup or reset. */
96 PRTREQ pPendingDummyReq;
97 /** Indicates whether PDMDrvHlpAsyncNotificationCompleted should be called by
98 * any of the dummy functions. */
99 bool volatile fDummySignal;
100 /** Release statistics: number of bytes written. */
101 STAMCOUNTER StatBytesWritten;
102 /** Release statistics: number of bytes read. */
103 STAMCOUNTER StatBytesRead;
104} DRVSCSI, *PDRVSCSI;
105
106/** Converts a pointer to DRVSCSI::ISCSIConnecotr to a PDRVSCSI. */
107#define PDMISCSICONNECTOR_2_DRVSCSI(pInterface) ( (PDRVSCSI)((uintptr_t)pInterface - RT_OFFSETOF(DRVSCSI, ISCSIConnector)) )
108
109#ifdef DEBUG
110/**
111 * Dumps a SCSI request structure for debugging purposes.
112 *
113 * @returns nothing.
114 * @param pRequest Pointer to the request to dump.
115 */
116static void drvscsiDumpScsiRequest(PPDMSCSIREQUEST pRequest)
117{
118 Log(("Dump for pRequest=%#p Command: %s\n", pRequest, SCSICmdText(pRequest->pbCDB[0])));
119 Log(("cbCDB=%u\n", pRequest->cbCDB));
120 for (uint32_t i = 0; i < pRequest->cbCDB; i++)
121 Log(("pbCDB[%u]=%#x\n", i, pRequest->pbCDB[i]));
122 Log(("cbScatterGather=%u\n", pRequest->cbScatterGather));
123 Log(("cScatterGatherEntries=%u\n", pRequest->cScatterGatherEntries));
124 /* Print all scatter gather entries. */
125 for (uint32_t i = 0; i < pRequest->cScatterGatherEntries; i++)
126 {
127 Log(("ScatterGatherEntry[%u].cbSeg=%u\n", i, pRequest->paScatterGatherHead[i].cbSeg));
128 Log(("ScatterGatherEntry[%u].pvSeg=%#p\n", i, pRequest->paScatterGatherHead[i].pvSeg));
129 }
130 Log(("pvUser=%#p\n", pRequest->pvUser));
131}
132#endif
133
134/**
135 * Copy the content of a buffer to a scatter gather list only
136 * copying only the amount of data which fits into the
137 * scatter gather list.
138 *
139 * @returns VBox status code.
140 * @param pRequest Pointer to the request which contains the S/G list entries.
141 * @param pvBuf Pointer to the buffer which should be copied.
142 * @param cbBuf Size of the buffer.
143 */
144static int drvscsiScatterGatherListCopyFromBuffer(PPDMSCSIREQUEST pRequest, void *pvBuf, size_t cbBuf)
145{
146 unsigned cSGEntry = 0;
147 PPDMDATASEG pSGEntry = &pRequest->paScatterGatherHead[cSGEntry];
148 uint8_t *pu8Buf = (uint8_t *)pvBuf;
149
150 LogFlowFunc(("pRequest=%#p pvBuf=%#p cbBuf=%u\n", pRequest, pvBuf, cbBuf));
151
152#ifdef DEBUG
153 for (unsigned i = 0; i < cbBuf; i++)
154 Log(("%s: pvBuf[%u]=%#x\n", __FUNCTION__, i, pu8Buf[i]));
155#endif
156
157 while (cSGEntry < pRequest->cScatterGatherEntries)
158 {
159 size_t cbToCopy = (cbBuf < pSGEntry->cbSeg) ? cbBuf : pSGEntry->cbSeg;
160
161 memcpy(pSGEntry->pvSeg, pu8Buf, cbToCopy);
162
163 cbBuf -= cbToCopy;
164 /* We finished. */
165 if (!cbBuf)
166 break;
167
168 /* Advance the buffer. */
169 pu8Buf += cbToCopy;
170
171 /* Go to the next entry in the list. */
172 pSGEntry++;
173 cSGEntry++;
174 }
175
176 return VINF_SUCCESS;
177}
178
179static void drvscsiPadStr(int8_t *pbDst, const char *pbSrc, uint32_t cbSize)
180{
181 for (uint32_t i = 0; i < cbSize; i++)
182 {
183 if (*pbSrc)
184 pbDst[i] = *pbSrc++;
185 else
186 pbDst[i] = ' ';
187 }
188}
189
190/**
191 * Set the sense and advanced sense key in the buffer for error conditions.
192 *
193 * @returns SCSI status code.
194 * @param pRequest Pointer to the request which contains the sense buffer.
195 * @param uSCSISenseKey The sense key to set.
196 * @param uSCSIASC The advanced sense key to set.
197 */
198DECLINLINE(int) drvscsiCmdError(PPDMSCSIREQUEST pRequest, uint8_t uSCSISenseKey, uint8_t uSCSIASC)
199{
200 AssertMsgReturn(pRequest->cbSenseBuffer >= 18, ("Sense buffer is not big enough\n"), SCSI_STATUS_OK);
201 AssertMsgReturn(pRequest->pbSenseBuffer, ("Sense buffer pointer is NULL\n"), SCSI_STATUS_OK);
202 memset(pRequest->pbSenseBuffer, 0, pRequest->cbSenseBuffer);
203 pRequest->pbSenseBuffer[0] = (1 << 7) | SCSI_SENSE_RESPONSE_CODE_CURR_FIXED; /* Fixed format */
204 pRequest->pbSenseBuffer[2] = uSCSISenseKey;
205 pRequest->pbSenseBuffer[7] = 10;
206 pRequest->pbSenseBuffer[12] = uSCSIASC;
207 pRequest->pbSenseBuffer[13] = 0x00; /** @todo: Provide more info. */
208 return SCSI_STATUS_CHECK_CONDITION;
209}
210
211/**
212 * Sets the sense key for a status good condition.
213 *
214 * @returns SCSI status code.
215 * @param pRequest Pointer to the request which contains the sense buffer.
216 */
217DECLINLINE(int) drvscsiCmdOk(PPDMSCSIREQUEST pRequest)
218{
219 if (pRequest->cbSenseBuffer == 0)
220 return SCSI_STATUS_OK;
221
222 AssertMsgReturn(pRequest->pbSenseBuffer, ("Sense buffer pointer is NULL\n"), SCSI_STATUS_OK);
223 memset(pRequest->pbSenseBuffer, 0, pRequest->cbSenseBuffer);
224 /*
225 * Setting this breaks Linux guests on the BusLogic controller.
226 * According to the SCSI SPC spec sense data is returned after a
227 * CHECK CONDITION status or a REQUEST SENSE command.
228 * Both SCSI controllers have a feature called Auto Sense which
229 * fetches the sense data automatically from the device
230 * with REQUEST SENSE. So the SCSI subsystem in Linux should
231 * find this sense data even if the command finishes successfully
232 * but if it finds valid sense data it will let the command fail
233 * and it doesn't detect attached disks anymore.
234 * Disabling makes it work again and no other guest shows errors
235 * so I will leave it disabled for now.
236 *
237 * On the other hand it is possible that the devices fetch the sense data
238 * only after a command failed so the content is really invalid if
239 * the command succeeds.
240 */
241#if 0
242 pRequest->pbSenseBuffer[0] = (1 << 7) | SCSI_SENSE_RESPONSE_CODE_CURR_FIXED; /* Fixed format */
243 pRequest->pbSenseBuffer[2] = SCSI_SENSE_NONE;
244 pRequest->pbSenseBuffer[7] = 10;
245 pRequest->pbSenseBuffer[12] = SCSI_ASC_NONE;
246 pRequest->pbSenseBuffer[13] = SCSI_ASC_NONE; /* Should be ASCQ but it has the same value for success. */
247#endif
248 return SCSI_STATUS_OK;
249}
250
251DECLINLINE(void) drvscsiH2BE_U16(uint8_t *pbBuf, uint16_t val)
252{
253 pbBuf[0] = val >> 8;
254 pbBuf[1] = val;
255}
256
257
258DECLINLINE(void) drvscsiH2BE_U24(uint8_t *pbBuf, uint32_t val)
259{
260 pbBuf[0] = val >> 16;
261 pbBuf[1] = val >> 8;
262 pbBuf[2] = val;
263}
264
265
266DECLINLINE(void) drvscsiH2BE_U32(uint8_t *pbBuf, uint32_t val)
267{
268 pbBuf[0] = val >> 24;
269 pbBuf[1] = val >> 16;
270 pbBuf[2] = val >> 8;
271 pbBuf[3] = val;
272}
273
274DECLINLINE(void) drvscsiH2BE_U64(uint8_t *pbBuf, uint64_t val)
275{
276 pbBuf[0] = val >> 56;
277 pbBuf[1] = val >> 48;
278 pbBuf[2] = val >> 40;
279 pbBuf[3] = val >> 32;
280 pbBuf[4] = val >> 24;
281 pbBuf[5] = val >> 16;
282 pbBuf[6] = val >> 8;
283 pbBuf[7] = val;
284}
285
286DECLINLINE(uint16_t) drvscsiBE2H_U16(const uint8_t *pbBuf)
287{
288 return (pbBuf[0] << 8) | pbBuf[1];
289}
290
291
292DECLINLINE(uint32_t) drvscsiBE2H_U24(const uint8_t *pbBuf)
293{
294 return (pbBuf[0] << 16) | (pbBuf[1] << 8) | pbBuf[2];
295}
296
297
298DECLINLINE(uint32_t) drvscsiBE2H_U32(const uint8_t *pbBuf)
299{
300 return (pbBuf[0] << 24) | (pbBuf[1] << 16) | (pbBuf[2] << 8) | pbBuf[3];
301}
302
303DECLINLINE(uint64_t) drvscsiBE2H_U64(const uint8_t *pbBuf)
304{
305 return ((uint64_t)pbBuf[0] << 56)
306 | ((uint64_t)pbBuf[1] << 48)
307 | ((uint64_t)pbBuf[2] << 40)
308 | ((uint64_t)pbBuf[3] << 32)
309 | ((uint64_t)pbBuf[4] << 24)
310 | ((uint64_t)pbBuf[5] << 16)
311 | ((uint64_t)pbBuf[6] << 8)
312 | (uint64_t)pbBuf[7];
313}
314
315/**
316 * Parses the CDB of a request and acts accordingly.
317 *
318 * @returns transfer direction type.
319 * @param pThis Pointer to the SCSI driver instance data.
320 * @param pRequest Pointer to the request to process.
321 * @param puOffset Where to store the start offset to start data transfer from.
322 * @param pcbToTransfer Where to store the number of bytes to transfer.
323 * @param piTxDir Where to store the data transfer direction.
324 */
325static int drvscsiProcessCDB(PDRVSCSI pThis, PPDMSCSIREQUEST pRequest, uint64_t *puOffset, uint32_t *pcbToTransfer, int *piTxDir)
326{
327 int iTxDir = PDMBLOCKTXDIR_NONE;
328 int rc = SCSI_STATUS_OK;
329
330 /* We check for a command which needs to be handled even for non existant LUNs. */
331 switch (pRequest->pbCDB[0])
332 {
333 case SCSI_INQUIRY:
334 {
335 SCSIINQUIRYDATA ScsiInquiryReply;
336
337 memset(&ScsiInquiryReply, 0, sizeof(ScsiInquiryReply));
338
339 ScsiInquiryReply.cbAdditional = 31;
340
341 /* We support only one attached device at LUN0 at the moment. */
342 if (pRequest->uLogicalUnit != 0)
343 {
344 ScsiInquiryReply.u5PeripheralDeviceType = SCSI_INQUIRY_DATA_PERIPHERAL_DEVICE_TYPE_UNKNOWN;
345 ScsiInquiryReply.u3PeripheralQualifier = SCSI_INQUIRY_DATA_PERIPHERAL_QUALIFIER_NOT_CONNECTED_NOT_SUPPORTED;
346 }
347 else
348 {
349 switch (pThis->enmType)
350 {
351 case PDMBLOCKTYPE_HARD_DISK:
352 ScsiInquiryReply.u5PeripheralDeviceType = SCSI_INQUIRY_DATA_PERIPHERAL_DEVICE_TYPE_DIRECT_ACCESS;
353 break;
354 default:
355 AssertMsgFailed(("Device type %u not supported\n", pThis->enmType));
356 }
357
358 ScsiInquiryReply.u3PeripheralQualifier = SCSI_INQUIRY_DATA_PERIPHERAL_QUALIFIER_CONNECTED;
359 ScsiInquiryReply.u3AnsiVersion = 0x05; /* SPC-4 compliant */
360 drvscsiPadStr(ScsiInquiryReply.achVendorId, "VBOX", 8);
361 drvscsiPadStr(ScsiInquiryReply.achProductId, "HARDDISK", 16);
362 drvscsiPadStr(ScsiInquiryReply.achProductLevel, "1.0", 4);
363 }
364
365 drvscsiScatterGatherListCopyFromBuffer(pRequest, &ScsiInquiryReply, sizeof(SCSIINQUIRYDATA));
366 rc = drvscsiCmdOk(pRequest);
367 break;
368 }
369 case SCSI_REPORT_LUNS:
370 {
371 /*
372 * If allocation length is less than 16 bytes SPC compliant devices have
373 * to return an error.
374 */
375 if (drvscsiBE2H_U32(&pRequest->pbCDB[6]) < 16)
376 rc = drvscsiCmdError(pRequest, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
377 else
378 {
379 uint8_t aReply[16]; /* We report only one LUN. */
380
381 memset(aReply, 0, sizeof(aReply));
382 drvscsiH2BE_U32(&aReply[0], 8); /* List length starts at position 0. */
383 drvscsiScatterGatherListCopyFromBuffer(pRequest, aReply, sizeof(aReply));
384 rc = drvscsiCmdOk(pRequest);
385 }
386 break;
387 }
388 case SCSI_TEST_UNIT_READY:
389 {
390 rc = drvscsiCmdOk(pRequest);
391 break;
392 }
393 default:
394 {
395 /* Now for commands which are only implemented for existant LUNs. */
396 if (RT_LIKELY(pRequest->uLogicalUnit == 0))
397 {
398 switch(pRequest->pbCDB[0])
399 {
400 case SCSI_READ_CAPACITY:
401 {
402 uint8_t aReply[8];
403 memset(aReply, 0, sizeof(aReply));
404
405 /*
406 * If sector size exceeds the maximum value that is
407 * able to be stored in 4 bytes return 0xffffffff in this field
408 */
409 if (pThis->cSectors > UINT32_C(0xffffffff))
410 drvscsiH2BE_U32(aReply, UINT32_C(0xffffffff));
411 else
412 drvscsiH2BE_U32(aReply, pThis->cSectors - 1);
413 drvscsiH2BE_U32(&aReply[4], 512);
414 drvscsiScatterGatherListCopyFromBuffer(pRequest, aReply, sizeof(aReply));
415 rc = drvscsiCmdOk(pRequest);
416 break;
417 }
418 case SCSI_MODE_SENSE_6:
419 {
420 uint8_t uModePage = pRequest->pbCDB[2] & 0x3f;
421 uint8_t aReply[24];
422 uint8_t *pu8ReplyPos;
423
424 memset(aReply, 0, sizeof(aReply));
425 aReply[0] = 4; /* Reply length 4. */
426 aReply[1] = 0; /* Default media type. */
427 aReply[2] = RT_BIT(4); /* Caching supported. */
428 aReply[3] = 0; /* Block descriptor length. */
429
430 pu8ReplyPos = aReply + 4;
431
432 if ((uModePage == 0x08) || (uModePage == 0x3f))
433 {
434 memset(pu8ReplyPos, 0, 20);
435 *pu8ReplyPos++ = 0x08; /* Page code. */
436 *pu8ReplyPos++ = 0x12; /* Size of the page. */
437 *pu8ReplyPos++ = 0x4; /* Write cache enabled. */
438 }
439
440 drvscsiScatterGatherListCopyFromBuffer(pRequest, aReply, sizeof(aReply));
441 rc = drvscsiCmdOk(pRequest);
442 break;
443 }
444 case SCSI_READ_6:
445 {
446 iTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
447 *puOffset = ((uint64_t) pRequest->pbCDB[3]
448 | (pRequest->pbCDB[2] << 8)
449 | ((pRequest->pbCDB[1] & 0x1f) << 16)) * 512;
450 *pcbToTransfer = ((uint32_t)pRequest->pbCDB[4]) * 512;
451 break;
452 }
453 case SCSI_READ_10:
454 {
455 iTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
456 *puOffset = ((uint64_t)drvscsiBE2H_U32(&pRequest->pbCDB[2])) * 512;
457 *pcbToTransfer = ((uint32_t)drvscsiBE2H_U16(&pRequest->pbCDB[7])) * 512;
458 break;
459 }
460 case SCSI_READ_12:
461 {
462 iTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
463 *puOffset = ((uint64_t)drvscsiBE2H_U32(&pRequest->pbCDB[2])) * 512;
464 *pcbToTransfer = ((uint32_t)drvscsiBE2H_U32(&pRequest->pbCDB[6])) * 512;
465 break;
466 }
467 case SCSI_READ_16:
468 {
469 iTxDir = PDMBLOCKTXDIR_FROM_DEVICE;
470 *puOffset = drvscsiBE2H_U64(&pRequest->pbCDB[2]) * 512;
471 *pcbToTransfer = ((uint32_t)drvscsiBE2H_U32(&pRequest->pbCDB[10])) * 512;
472 break;
473 }
474 case SCSI_WRITE_6:
475 {
476 iTxDir = PDMBLOCKTXDIR_TO_DEVICE;
477 *puOffset = ((uint64_t) pRequest->pbCDB[3]
478 | (pRequest->pbCDB[2] << 8)
479 | ((pRequest->pbCDB[1] & 0x1f) << 16)) * 512;
480 *pcbToTransfer = ((uint32_t)pRequest->pbCDB[4]) * 512;
481 break;
482 }
483 case SCSI_WRITE_10:
484 {
485 iTxDir = PDMBLOCKTXDIR_TO_DEVICE;
486 *puOffset = ((uint64_t)drvscsiBE2H_U32(&pRequest->pbCDB[2])) * 512;
487 *pcbToTransfer = ((uint32_t)drvscsiBE2H_U16(&pRequest->pbCDB[7])) * 512;
488 break;
489 }
490 case SCSI_WRITE_12:
491 {
492 iTxDir = PDMBLOCKTXDIR_TO_DEVICE;
493 *puOffset = ((uint64_t)drvscsiBE2H_U32(&pRequest->pbCDB[2])) * 512;
494 *pcbToTransfer = ((uint32_t)drvscsiBE2H_U32(&pRequest->pbCDB[6])) * 512;
495 break;
496 }
497 case SCSI_WRITE_16:
498 {
499 iTxDir = PDMBLOCKTXDIR_TO_DEVICE;
500 *puOffset = drvscsiBE2H_U64(&pRequest->pbCDB[2]) * 512;
501 *pcbToTransfer = ((uint32_t)drvscsiBE2H_U32(&pRequest->pbCDB[10])) * 512;
502 break;
503 }
504 case SCSI_SYNCHRONIZE_CACHE:
505 {
506 /* @todo When async mode implemented we have to move this out here. */
507 int rc2 = pThis->pDrvBlock->pfnFlush(pThis->pDrvBlock);
508 AssertMsgRC(rc2, ("Flushing data failed rc=%Rrc\n", rc2));
509 break;
510 }
511 case SCSI_READ_BUFFER:
512 {
513 uint8_t uDataMode = pRequest->pbCDB[1] & 0x1f;
514
515 switch (uDataMode)
516 {
517 case 0x00:
518 case 0x01:
519 case 0x02:
520 case 0x03:
521 case 0x0a:
522 break;
523 case 0x0b:
524 {
525 uint8_t aReply[4];
526
527 /* We do not implement an echo buffer. */
528 memset(aReply, 0, sizeof(aReply));
529
530 drvscsiScatterGatherListCopyFromBuffer(pRequest, aReply, sizeof(aReply));
531 rc = drvscsiCmdOk(pRequest);
532 break;
533 }
534 case 0x1a:
535 case 0x1c:
536 break;
537 default:
538 AssertMsgFailed(("Invalid data mode\n"));
539 }
540 break;
541 }
542 case SCSI_START_STOP_UNIT:
543 {
544 /* Nothing to do. */
545 break;
546 }
547 case SCSI_LOG_SENSE:
548 {
549 uint16_t cbMax = drvscsiBE2H_U16(&pRequest->pbCDB[7]);
550 uint8_t uPageCode = pRequest->pbCDB[2] & 0x3f;
551 uint8_t uSubPageCode = pRequest->pbCDB[3];
552
553 switch (uPageCode)
554 {
555 case 0x00:
556 {
557 if (uSubPageCode == 0)
558 {
559 uint8_t aReply[4];
560
561 aReply[0] = 0;
562 aReply[1] = 0;
563 aReply[2] = 0;
564 aReply[3] = 0;
565 drvscsiScatterGatherListCopyFromBuffer(pRequest, aReply, sizeof(aReply));
566 rc = drvscsiCmdOk(pRequest);
567 break;
568 }
569 }
570 default:
571 rc = drvscsiCmdError(pRequest, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
572 }
573 break;
574 }
575 case SCSI_SERVICE_ACTION_IN_16:
576 {
577 switch (pRequest->pbCDB[1] & 0x1f)
578 {
579 case SCSI_SVC_ACTION_IN_READ_CAPACITY_16:
580 {
581 uint8_t aReply[32];
582
583 memset(aReply, 0, sizeof(aReply));
584 drvscsiH2BE_U64(aReply, pThis->cSectors - 1);
585 drvscsiH2BE_U32(&aReply[8], 512);
586 /* Leave the rest 0 */
587
588 drvscsiScatterGatherListCopyFromBuffer(pRequest, aReply, sizeof(aReply));
589 rc = drvscsiCmdOk(pRequest);
590 break;
591 }
592 default:
593 rc = drvscsiCmdError(pRequest, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET); /* Don't know if this is correct */
594 }
595 break;
596 }
597 default:
598 //AssertMsgFailed(("Command %#x [%s] not implemented\n", pRequest->pbCDB[0], SCSICmdText(pRequest->pbCDB[0])));
599 rc = drvscsiCmdError(pRequest, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE);
600 }
601 }
602 else
603 {
604 /* Report an error. */
605 rc = drvscsiCmdError(pRequest, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_LOGICAL_UNIT_DOES_NOT_RESPOND_TO_SELECTION);
606 }
607 break;
608 }
609 }
610
611 *piTxDir = iTxDir;
612
613 return rc;
614}
615
616static int drvscsiProcessRequestOne(PDRVSCSI pThis, PPDMSCSIREQUEST pRequest)
617{
618 int rc = VINF_SUCCESS;
619 int iTxDir;
620 int rcCompletion;
621 uint64_t uOffset = UINT64_MAX; /* initialized to shut up gcc warnings. */
622 uint32_t cbToTransfer = UINT32_MAX; /* ditto */
623 uint32_t cSegmentsLeft = UINT32_MAX; /* ditto */
624
625 LogFlowFunc(("Entered\n"));
626
627#ifdef DEBUG
628 drvscsiDumpScsiRequest(pRequest);
629#endif
630 rcCompletion = drvscsiProcessCDB(pThis, pRequest, &uOffset, &cbToTransfer, &iTxDir);
631 if ((rcCompletion == SCSI_STATUS_OK) && (iTxDir != PDMBLOCKTXDIR_NONE))
632 {
633 PPDMDATASEG pSegActual;
634
635 pSegActual = &pRequest->paScatterGatherHead[0];
636 cSegmentsLeft = pRequest->cScatterGatherEntries;
637
638 while(cbToTransfer && cSegmentsLeft)
639 {
640 uint32_t cbProcess = (cbToTransfer < pSegActual->cbSeg) ? cbToTransfer : (uint32_t)pSegActual->cbSeg;
641
642 Log(("%s: uOffset=%llu cbToTransfer=%u\n", __FUNCTION__, uOffset, cbToTransfer));
643
644 if (iTxDir == PDMBLOCKTXDIR_FROM_DEVICE)
645 {
646 pThis->pLed->Asserted.s.fReading = pThis->pLed->Actual.s.fReading = 1;
647 rc = pThis->pDrvBlock->pfnRead(pThis->pDrvBlock, uOffset,
648 pSegActual->pvSeg, cbProcess);
649 pThis->pLed->Actual.s.fReading = 0;
650 if (RT_FAILURE(rc))
651 AssertMsgFailed(("%s: Failed to read data %Rrc\n", __FUNCTION__, rc));
652 STAM_REL_COUNTER_ADD(&pThis->StatBytesRead, cbProcess);
653 }
654 else
655 {
656 pThis->pLed->Asserted.s.fWriting = pThis->pLed->Actual.s.fWriting = 1;
657 rc = pThis->pDrvBlock->pfnWrite(pThis->pDrvBlock, uOffset,
658 pSegActual->pvSeg, cbProcess);
659 pThis->pLed->Actual.s.fWriting = 0;
660 if (RT_FAILURE(rc))
661 AssertMsgFailed(("%s: Failed to write data %Rrc\n", __FUNCTION__, rc));
662 STAM_REL_COUNTER_ADD(&pThis->StatBytesWritten, cbProcess);
663 }
664
665 /* Go to the next entry. */
666 uOffset += cbProcess;
667 cbToTransfer -= cbProcess;
668 pSegActual++;
669 cSegmentsLeft--;
670 }
671 AssertMsg(!cbToTransfer && !cSegmentsLeft,
672 ("Transfer incomplete cbToTransfer=%u cSegmentsLeft=%u\n", cbToTransfer, cSegmentsLeft));
673 drvscsiCmdOk(pRequest);
674 }
675
676 /* Notify device. */
677 rc = pThis->pDevScsiPort->pfnSCSIRequestCompleted(pThis->pDevScsiPort, pRequest, rcCompletion);
678 AssertMsgRC(rc, ("Error while notifying device rc=%Rrc\n", rc));
679
680 return rc;
681}
682
683/**
684 * Dummy request function used by drvscsiReset to wait for all pending requests
685 * to complete prior to the device reset.
686 *
687 * @param pThis Pointer to the instace data.
688 * @returns VINF_SUCCESS.
689 */
690static int drvscsiAsyncIOLoopSyncCallback(PDRVSCSI pThis)
691{
692 if (pThis->fDummySignal)
693 PDMDrvHlpAsyncNotificationCompleted(pThis->pDrvIns);
694 return VINF_SUCCESS;
695}
696
697/**
698 * Request function to wakeup the thread.
699 *
700 * @param pThis Pointer to the instace data.
701 * @returns VWRN_STATE_CHANGED.
702 */
703static int drvscsiAsyncIOLoopWakeupFunc(PDRVSCSI pThis)
704{
705 if (pThis->fDummySignal)
706 PDMDrvHlpAsyncNotificationCompleted(pThis->pDrvIns);
707 return VWRN_STATE_CHANGED;
708}
709
710/**
711 * The thread function which processes the requests asynchronously.
712 *
713 * @returns VBox status code.
714 * @param pDrvIns Pointer to the driver instance data.
715 * @param pThread Pointer to the thread instance data.
716 */
717static int drvscsiAsyncIOLoop(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
718{
719 int rc = VINF_SUCCESS;
720 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
721
722 LogFlowFunc(("Entering async IO loop.\n"));
723
724 if (pThread->enmState == PDMTHREADSTATE_INITIALIZING)
725 return VINF_SUCCESS;
726
727 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
728 {
729 rc = RTReqProcess(pThis->pQueueRequests, RT_INDEFINITE_WAIT);
730 AssertMsg(rc == VWRN_STATE_CHANGED, ("Left RTReqProcess and error code is not VWRN_STATE_CHANGED rc=%Rrc\n", rc));
731 }
732
733 return VINF_SUCCESS;
734}
735
736/**
737 * Deals with any pending dummy request
738 *
739 * @returns true if no pending dummy request, false if still pending.
740 * @param pThis The instance data.
741 * @param cMillies The number of milliseconds to wait for any
742 * pending request to finish.
743 */
744static bool drvscsiAsyncIOLoopNoPendingDummy(PDRVSCSI pThis, uint32_t cMillies)
745{
746 if (!pThis->pPendingDummyReq)
747 return true;
748 int rc = RTReqWait(pThis->pPendingDummyReq, cMillies);
749 if (RT_FAILURE(rc))
750 return false;
751 RTReqFree(pThis->pPendingDummyReq);
752 pThis->pPendingDummyReq = NULL;
753 return true;
754}
755
756static int drvscsiAsyncIOLoopWakeup(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
757{
758 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
759 PRTREQ pReq;
760 int rc;
761
762 AssertMsgReturn(pThis->pQueueRequests, ("pQueueRequests is NULL\n"), VERR_INVALID_STATE);
763
764 if (!drvscsiAsyncIOLoopNoPendingDummy(pThis, 10000 /* 10 sec */))
765 {
766 LogRel(("drvscsiAsyncIOLoopWakeup#%u: previous dummy request is still pending\n", pDrvIns->iInstance));
767 return VERR_TIMEOUT;
768 }
769
770 rc = RTReqCall(pThis->pQueueRequests, &pReq, 10000 /* 10 sec. */, (PFNRT)drvscsiAsyncIOLoopWakeupFunc, 1, pThis);
771 if (RT_SUCCESS(rc))
772 RTReqFree(pReq);
773 else
774 {
775 pThis->pPendingDummyReq = pReq;
776 LogRel(("drvscsiAsyncIOLoopWakeup#%u: %Rrc pReq=%p\n", pDrvIns->iInstance, rc, pReq));
777 }
778
779 return rc;
780}
781
782/* -=-=-=-=- ISCSIConnector -=-=-=-=- */
783
784/** @copydoc PDMISCSICONNECTOR::pfnSCSIRequestSend. */
785static DECLCALLBACK(int) drvscsiRequestSend(PPDMISCSICONNECTOR pInterface, PPDMSCSIREQUEST pSCSIRequest)
786{
787 int rc;
788 PDRVSCSI pThis = PDMISCSICONNECTOR_2_DRVSCSI(pInterface);
789 PRTREQ pReq;
790
791 AssertMsgReturn(pThis->pQueueRequests, ("pQueueRequests is NULL\n"), VERR_INVALID_STATE);
792
793 rc = RTReqCallEx(pThis->pQueueRequests, &pReq, 0, RTREQFLAGS_NO_WAIT, (PFNRT)drvscsiProcessRequestOne, 2, pThis, pSCSIRequest);
794 AssertMsgReturn(RT_SUCCESS(rc), ("Inserting request into queue failed rc=%Rrc\n", rc), rc);
795
796 return VINF_SUCCESS;
797}
798
799/* -=-=-=-=- IBase -=-=-=-=- */
800
801/**
802 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
803 */
804static DECLCALLBACK(void *) drvscsiQueryInterface(PPDMIBASE pInterface, const char *pszIID)
805{
806 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
807 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
808
809 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
810 PDMIBASE_RETURN_INTERFACE(pszIID, PDMISCSICONNECTOR, &pThis->ISCSIConnector);
811 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBLOCKPORT, &pThis->IPort);
812 return NULL;
813}
814
815/**
816 * Worker for drvscsiReset, drvscsiSuspend and drvscsiPowerOff.
817 *
818 * @param pDrvIns The driver instance.
819 * @param pfnAsyncNotify The async callback.
820 */
821static void drvscsiR3ResetOrSuspendOrPowerOff(PPDMDRVINS pDrvIns, PFNPDMDRVASYNCNOTIFY pfnAsyncNotify)
822{
823 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
824
825 if (!pThis->pQueueRequests)
826 return;
827
828 ASMAtomicWriteBool(&pThis->fDummySignal, true);
829 if (drvscsiAsyncIOLoopNoPendingDummy(pThis, 0 /*ms*/))
830 {
831 if (!RTReqIsBusy(pThis->pQueueRequests))
832 {
833 ASMAtomicWriteBool(&pThis->fDummySignal, false);
834 return;
835 }
836
837 PRTREQ pReq;
838 int rc = RTReqCall(pThis->pQueueRequests, &pReq, 0 /*ms*/, (PFNRT)drvscsiAsyncIOLoopSyncCallback, 1, pThis);
839 if (RT_SUCCESS(rc))
840 {
841 ASMAtomicWriteBool(&pThis->fDummySignal, false);
842 RTReqFree(pReq);
843 return;
844 }
845
846 pThis->pPendingDummyReq = pReq;
847 }
848 PDMDrvHlpSetAsyncNotification(pDrvIns, pfnAsyncNotify);
849}
850
851/**
852 * Callback employed by drvscsiSuspend and drvscsiPowerOff.
853 *
854 * @returns true if we've quiesced, false if we're still working.
855 * @param pDrvIns The driver instance.
856 */
857static DECLCALLBACK(bool) drvscsiIsAsyncSuspendOrPowerOffDone(PPDMDRVINS pDrvIns)
858{
859 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
860
861 if (!drvscsiAsyncIOLoopNoPendingDummy(pThis, 0 /*ms*/))
862 return false;
863 ASMAtomicWriteBool(&pThis->fDummySignal, false);
864 PDMR3ThreadSuspend(pThis->pAsyncIOThread);
865 return true;
866}
867
868/**
869 * @copydoc FNPDMDRVPOWEROFF
870 */
871static DECLCALLBACK(void) drvscsiPowerOff(PPDMDRVINS pDrvIns)
872{
873 drvscsiR3ResetOrSuspendOrPowerOff(pDrvIns, drvscsiIsAsyncSuspendOrPowerOffDone);
874}
875
876/**
877 * @copydoc FNPDMDRVSUSPEND
878 */
879static DECLCALLBACK(void) drvscsiSuspend(PPDMDRVINS pDrvIns)
880{
881 drvscsiR3ResetOrSuspendOrPowerOff(pDrvIns, drvscsiIsAsyncSuspendOrPowerOffDone);
882}
883
884/**
885 * Callback employed by drvscsiReset.
886 *
887 * @returns true if we've quiesced, false if we're still working.
888 * @param pDrvIns The driver instance.
889 */
890static DECLCALLBACK(bool) drvscsiIsAsyncResetDone(PPDMDRVINS pDrvIns)
891{
892 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
893
894 if (!drvscsiAsyncIOLoopNoPendingDummy(pThis, 0 /*ms*/))
895 return false;
896 ASMAtomicWriteBool(&pThis->fDummySignal, false);
897 return true;
898}
899
900/**
901 * @copydoc FNPDMDRVRESET
902 */
903static DECLCALLBACK(void) drvscsiReset(PPDMDRVINS pDrvIns)
904{
905 drvscsiR3ResetOrSuspendOrPowerOff(pDrvIns, drvscsiIsAsyncResetDone);
906}
907
908/**
909 * Destruct a driver instance.
910 *
911 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
912 * resources can be freed correctly.
913 *
914 * @param pDrvIns The driver instance data.
915 */
916static DECLCALLBACK(void) drvscsiDestruct(PPDMDRVINS pDrvIns)
917{
918 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
919 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
920
921 if (pThis->pQueueRequests)
922 {
923 if (!drvscsiAsyncIOLoopNoPendingDummy(pThis, 100 /*ms*/))
924 LogRel(("drvscsiDestruct#%u: previous dummy request is still pending\n", pDrvIns->iInstance));
925
926 int rc = RTReqDestroyQueue(pThis->pQueueRequests);
927 AssertMsgRC(rc, ("Failed to destroy queue rc=%Rrc\n", rc));
928 }
929}
930
931/**
932 * Construct a block driver instance.
933 *
934 * @copydoc FNPDMDRVCONSTRUCT
935 */
936static DECLCALLBACK(int) drvscsiConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
937{
938 PDRVSCSI pThis = PDMINS_2_DATA(pDrvIns, PDRVSCSI);
939 LogFlowFunc(("pDrvIns=%#p pCfg=%#p\n", pDrvIns, pCfg));
940 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
941
942 /*
943 * Initialize the instance data.
944 */
945 pThis->pDrvIns = pDrvIns;
946 pThis->ISCSIConnector.pfnSCSIRequestSend = drvscsiRequestSend;
947
948 pDrvIns->IBase.pfnQueryInterface = drvscsiQueryInterface;
949
950 /*
951 * Try attach driver below and query it's block interface.
952 */
953 int rc = PDMDrvHlpAttach(pDrvIns, fFlags, &pThis->pDrvBase);
954 AssertMsgReturn(RT_SUCCESS(rc), ("Attaching driver below failed rc=%Rrc\n", rc), rc);
955
956 /*
957 * Query the block and blockbios interfaces.
958 */
959 pThis->pDrvBlock = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBLOCK);
960 if (!pThis->pDrvBlock)
961 {
962 AssertMsgFailed(("Configuration error: No block interface!\n"));
963 return VERR_PDM_MISSING_INTERFACE;
964 }
965 pThis->pDrvBlockBios = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBLOCKBIOS);
966 if (!pThis->pDrvBlockBios)
967 {
968 AssertMsgFailed(("Configuration error: No block BIOS interface!\n"));
969 return VERR_PDM_MISSING_INTERFACE;
970 }
971
972 /* Query the SCSI port interface above. */
973 pThis->pDevScsiPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMISCSIPORT);
974 AssertMsgReturn(pThis->pDevScsiPort, ("Missing SCSI port interface above\n"), VERR_PDM_MISSING_INTERFACE);
975
976 pThis->pDrvMount = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIMOUNT);
977
978 /* Query the optional LED interface above. */
979 pThis->pLedPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMILEDPORTS);
980 if (pThis->pLedPort != NULL)
981 {
982 /* Get The Led. */
983 rc = pThis->pLedPort->pfnQueryStatusLed(pThis->pLedPort, 0, &pThis->pLed);
984 if (RT_FAILURE(rc))
985 pThis->pLed = &pThis->Led;
986 }
987 else
988 pThis->pLed = &pThis->Led;
989
990 /* Try to get the optional async block interface. */
991 pThis->pDrvBlockAsync = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBLOCKASYNC);
992
993 PDMBLOCKTYPE enmType = pThis->pDrvBlock->pfnGetType(pThis->pDrvBlock);
994 if (enmType != PDMBLOCKTYPE_HARD_DISK)
995 return PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_UNSUPPORTED_BLOCK_TYPE, RT_SRC_POS,
996 N_("Only hard disks are currently supported as SCSI devices (enmType=%d)"),
997 enmType);
998 pThis->enmType = enmType;
999 pThis->cSectors = pThis->pDrvBlock->pfnGetSize(pThis->pDrvBlock) / 512;
1000
1001 /* Create request queue. */
1002 rc = RTReqCreateQueue(&pThis->pQueueRequests);
1003 AssertMsgReturn(RT_SUCCESS(rc), ("Failed to create request queue rc=%Rrc\n"), rc);
1004
1005 /* Register statistics counter. */
1006 /** @todo aeichner: Find a way to put the instance number of the attached
1007 * controller device when we support more than one controller of the same type.
1008 * At the moment we have the 0 hardcoded. */
1009 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatBytesRead, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
1010 "Amount of data read.", "/Devices/SCSI0/%d/ReadBytes", pDrvIns->iInstance);
1011 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatBytesWritten, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
1012 "Amount of data written.", "/Devices/SCSI0/%d/WrittenBytes", pDrvIns->iInstance);
1013
1014 /* Create I/O thread. */
1015 rc = PDMDrvHlpPDMThreadCreate(pDrvIns, &pThis->pAsyncIOThread, pThis, drvscsiAsyncIOLoop,
1016 drvscsiAsyncIOLoopWakeup, 0, RTTHREADTYPE_IO, "SCSI async IO");
1017 AssertMsgReturn(RT_SUCCESS(rc), ("Failed to create async I/O thread rc=%Rrc\n"), rc);
1018
1019 return VINF_SUCCESS;
1020}
1021
1022/**
1023 * SCSI driver registration record.
1024 */
1025const PDMDRVREG g_DrvSCSI =
1026{
1027 /* u32Version */
1028 PDM_DRVREG_VERSION,
1029 /* szName */
1030 "SCSI",
1031 /* szRCMod */
1032 "",
1033 /* szR0Mod */
1034 "",
1035 /* pszDescription */
1036 "Generic SCSI driver.",
1037 /* fFlags */
1038 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1039 /* fClass. */
1040 PDM_DRVREG_CLASS_SCSI,
1041 /* cMaxInstances */
1042 ~0,
1043 /* cbInstance */
1044 sizeof(DRVSCSI),
1045 /* pfnConstruct */
1046 drvscsiConstruct,
1047 /* pfnDestruct */
1048 drvscsiDestruct,
1049 /* pfnRelocate */
1050 NULL,
1051 /* pfnIOCtl */
1052 NULL,
1053 /* pfnPowerOn */
1054 NULL,
1055 /* pfnReset */
1056 drvscsiReset,
1057 /* pfnSuspend */
1058 drvscsiSuspend,
1059 /* pfnResume */
1060 NULL,
1061 /* pfnAttach */
1062 NULL,
1063 /* pfnDetach */
1064 NULL,
1065 /* pfnPowerOff */
1066 drvscsiPowerOff,
1067 /* pfnSoftReset */
1068 NULL,
1069 /* u32EndVersion */
1070 PDM_DRVREG_VERSION
1071};
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