VirtualBox

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

Last change on this file since 22925 was 22850, checked in by vboxsync, 15 years ago

SCSI: Remove non supported command reporting it as an invalid opcode instead. Fixes botting Fedora 11 after kernel update

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