VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvHostDVD.cpp@ 64744

Last change on this file since 64744 was 64413, checked in by vboxsync, 8 years ago

Devices/Storage/HostDVD: Fix sense reporting, don't call drvHostDvdCmdError() after the command was passed through because the sense buffer is already valid and would be zeroed out, fixes at least a Windows hang when trying to access an audio CD

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 31.1 KB
Line 
1/* $Id: DrvHostDVD.cpp 64413 2016-10-25 13:07:55Z vboxsync $ */
2/** @file
3 * DrvHostDVD - Host DVD block driver.
4 */
5
6/*
7 * Copyright (C) 2006-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_DRV_HOST_DVD
23#include <iprt/asm.h>
24#include <VBox/vmm/pdmdrv.h>
25#include <VBox/vmm/pdmstorageifs.h>
26#include <iprt/asm.h>
27#include <iprt/assert.h>
28#include <iprt/file.h>
29#include <iprt/string.h>
30#include <iprt/thread.h>
31#include <iprt/critsect.h>
32#include <VBox/scsi.h>
33
34#include "VBoxDD.h"
35#include "DrvHostBase.h"
36#include "ATAPIPassthrough.h"
37
38/** ATAPI sense info size. */
39#define ATAPI_SENSE_SIZE 64
40/** Size of an ATAPI packet. */
41#define ATAPI_PACKET_SIZE 12
42
43/**
44 * Host DVD driver instance data.
45 */
46typedef struct DRVHOSTDVD
47{
48 /** Base drivr data. */
49 DRVHOSTBASE Core;
50 /** The current tracklist of the loaded medium if passthrough is used. */
51 PTRACKLIST pTrackList;
52 /** ATAPI sense data. */
53 uint8_t abATAPISense[ATAPI_SENSE_SIZE];
54} DRVHOSTDVD;
55/** Pointer to the host DVD driver instance data. */
56typedef DRVHOSTDVD *PDRVHOSTDVD;
57
58/*********************************************************************************************************************************
59* Internal Functions *
60*********************************************************************************************************************************/
61
62DECLINLINE(void) drvHostDvdH2BE_U16(uint8_t *pbBuf, uint16_t val)
63{
64 pbBuf[0] = val >> 8;
65 pbBuf[1] = val;
66}
67
68
69DECLINLINE(void) drvHostDvdH2BE_U24(uint8_t *pbBuf, uint32_t val)
70{
71 pbBuf[0] = val >> 16;
72 pbBuf[1] = val >> 8;
73 pbBuf[2] = val;
74}
75
76
77DECLINLINE(void) drvHostDvdH2BE_U32(uint8_t *pbBuf, uint32_t val)
78{
79 pbBuf[0] = val >> 24;
80 pbBuf[1] = val >> 16;
81 pbBuf[2] = val >> 8;
82 pbBuf[3] = val;
83}
84
85
86DECLINLINE(uint16_t) drvHostDvdBE2H_U16(const uint8_t *pbBuf)
87{
88 return (pbBuf[0] << 8) | pbBuf[1];
89}
90
91
92DECLINLINE(uint32_t) drvHostDvdBE2H_U24(const uint8_t *pbBuf)
93{
94 return (pbBuf[0] << 16) | (pbBuf[1] << 8) | pbBuf[2];
95}
96
97
98DECLINLINE(uint32_t) drvHostDvdBE2H_U32(const uint8_t *pbBuf)
99{
100 return (pbBuf[0] << 24) | (pbBuf[1] << 16) | (pbBuf[2] << 8) | pbBuf[3];
101}
102
103
104DECLINLINE(void) drvHostDvdLBA2MSF(uint8_t *pbBuf, uint32_t iATAPILBA)
105{
106 iATAPILBA += 150;
107 pbBuf[0] = (iATAPILBA / 75) / 60;
108 pbBuf[1] = (iATAPILBA / 75) % 60;
109 pbBuf[2] = iATAPILBA % 75;
110}
111
112
113DECLINLINE(uint32_t) drvHostDvdMSF2LBA(const uint8_t *pbBuf)
114{
115 return (pbBuf[0] * 60 + pbBuf[1]) * 75 + pbBuf[2];
116}
117
118static uint8_t drvHostDvdCmdOK(PDRVHOSTDVD pThis)
119{
120 memset(pThis->abATAPISense, '\0', sizeof(pThis->abATAPISense));
121 pThis->abATAPISense[0] = 0x70;
122 pThis->abATAPISense[7] = 10;
123 return SCSI_STATUS_OK;
124}
125
126static uint8_t drvHostDvdCmdError(PDRVHOSTDVD pThis, const uint8_t *pabATAPISense, size_t cbATAPISense)
127{
128 Log(("%s: sense=%#x (%s) asc=%#x ascq=%#x (%s)\n", __FUNCTION__, pabATAPISense[2] & 0x0f, SCSISenseText(pabATAPISense[2] & 0x0f),
129 pabATAPISense[12], pabATAPISense[13], SCSISenseExtText(pabATAPISense[12], pabATAPISense[13])));
130 memset(pThis->abATAPISense, '\0', sizeof(pThis->abATAPISense));
131 memcpy(pThis->abATAPISense, pabATAPISense, RT_MIN(cbATAPISense, sizeof(pThis->abATAPISense)));
132 return SCSI_STATUS_CHECK_CONDITION;
133}
134
135/** @todo deprecated function - doesn't provide enough info. Replace by direct
136 * calls to drvHostDvdCmdError() with full data. */
137static uint8_t drvHostDvdCmdErrorSimple(PDRVHOSTDVD pThis, uint8_t uATAPISenseKey, uint8_t uATAPIASC)
138{
139 uint8_t abATAPISense[ATAPI_SENSE_SIZE];
140 memset(abATAPISense, '\0', sizeof(abATAPISense));
141 abATAPISense[0] = 0x70 | (1 << 7);
142 abATAPISense[2] = uATAPISenseKey & 0x0f;
143 abATAPISense[7] = 10;
144 abATAPISense[12] = uATAPIASC;
145 return drvHostDvdCmdError(pThis, abATAPISense, sizeof(abATAPISense));
146}
147
148static void drvHostDvdSCSIPadStr(uint8_t *pbDst, const char *pbSrc, uint32_t cbSize)
149{
150 for (uint32_t i = 0; i < cbSize; i++)
151 {
152 if (*pbSrc)
153 pbDst[i] = *pbSrc++;
154 else
155 pbDst[i] = ' ';
156 }
157}
158
159
160static bool drvHostDvdParseCdb(PDRVHOSTDVD pThis, PDRVHOSTBASEREQ pReq,
161 const uint8_t *pbCdb, size_t cbCdb, size_t cbBuf,
162 PDMMEDIATXDIR *penmTxDir, size_t *pcbXfer,
163 size_t *pcbSector, uint8_t *pu8ScsiSts)
164{
165 uint32_t uLba = 0;
166 uint32_t cSectors = 0;
167 size_t cbSector = 0;
168 size_t cbXfer = 0;
169 bool fPassthrough = false;
170 PDMMEDIATXDIR enmTxDir = PDMMEDIATXDIR_NONE;
171
172 RT_NOREF(cbCdb);
173
174 switch (pbCdb[0])
175 {
176 /* First the commands we can pass through without further processing. */
177 case SCSI_BLANK:
178 case SCSI_CLOSE_TRACK_SESSION:
179 case SCSI_LOAD_UNLOAD_MEDIUM:
180 case SCSI_PAUSE_RESUME:
181 case SCSI_PLAY_AUDIO_10:
182 case SCSI_PLAY_AUDIO_12:
183 case SCSI_PLAY_AUDIO_MSF:
184 case SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL:
185 case SCSI_REPAIR_TRACK:
186 case SCSI_RESERVE_TRACK:
187 case SCSI_SCAN:
188 case SCSI_SEEK_10:
189 case SCSI_SET_CD_SPEED:
190 case SCSI_SET_READ_AHEAD:
191 case SCSI_START_STOP_UNIT:
192 case SCSI_STOP_PLAY_SCAN:
193 case SCSI_SYNCHRONIZE_CACHE:
194 case SCSI_TEST_UNIT_READY:
195 case SCSI_VERIFY_10:
196 fPassthrough = true;
197 break;
198 case SCSI_ERASE_10:
199 uLba = drvHostDvdBE2H_U32(pbCdb + 2);
200 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
201 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
202 fPassthrough = true;
203 break;
204 case SCSI_FORMAT_UNIT:
205 cbXfer = cbBuf;
206 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
207 fPassthrough = true;
208 break;
209 case SCSI_GET_CONFIGURATION:
210 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
211 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
212 fPassthrough = true;
213 break;
214 case SCSI_GET_EVENT_STATUS_NOTIFICATION:
215 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
216 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
217 fPassthrough = true;
218 break;
219 case SCSI_GET_PERFORMANCE:
220 cbXfer = cbBuf;
221 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
222 fPassthrough = true;
223 break;
224 case SCSI_INQUIRY:
225 cbXfer = drvHostDvdBE2H_U16(pbCdb + 3);
226 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
227 fPassthrough = true;
228 break;
229 case SCSI_MECHANISM_STATUS:
230 cbXfer = drvHostDvdBE2H_U16(pbCdb + 8);
231 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
232 fPassthrough = true;
233 break;
234 case SCSI_MODE_SELECT_10:
235 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
236 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
237 fPassthrough = true;
238 break;
239 case SCSI_MODE_SENSE_10:
240 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
241 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
242 fPassthrough = true;
243 break;
244 case SCSI_READ_10:
245 uLba = drvHostDvdBE2H_U32(pbCdb + 2);
246 cSectors = drvHostDvdBE2H_U16(pbCdb + 7);
247 cbSector = 2048;
248 cbXfer = cSectors * cbSector;
249 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
250 fPassthrough = true;
251 break;
252 case SCSI_READ_12:
253 uLba = drvHostDvdBE2H_U32(pbCdb + 2);
254 cSectors = drvHostDvdBE2H_U32(pbCdb + 6);
255 cbSector = 2048;
256 cbXfer = cSectors * cbSector;
257 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
258 fPassthrough = true;
259 break;
260 case SCSI_READ_BUFFER:
261 cbXfer = drvHostDvdBE2H_U24(pbCdb + 6);
262 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
263 fPassthrough = true;
264 break;
265 case SCSI_READ_BUFFER_CAPACITY:
266 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
267 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
268 fPassthrough = true;
269 break;
270 case SCSI_READ_CAPACITY:
271 cbXfer = 8;
272 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
273 fPassthrough = true;
274 break;
275 case SCSI_READ_CD:
276 case SCSI_READ_CD_MSF:
277 {
278 /* Get sector size based on the expected sector type field. */
279 switch ((pbCdb[1] >> 2) & 0x7)
280 {
281 case 0x0: /* All types. */
282 {
283 uint32_t iLbaStart;
284
285 if (pbCdb[0] == SCSI_READ_CD)
286 iLbaStart = drvHostDvdBE2H_U32(&pbCdb[2]);
287 else
288 iLbaStart = drvHostDvdMSF2LBA(&pbCdb[3]);
289
290 if (pThis->pTrackList)
291 cbSector = ATAPIPassthroughTrackListGetSectorSizeFromLba(pThis->pTrackList, iLbaStart);
292 else
293 cbSector = 2048; /* Might be incorrect if we couldn't determine the type. */
294 break;
295 }
296 case 0x1: /* CD-DA */
297 cbSector = 2352;
298 break;
299 case 0x2: /* Mode 1 */
300 cbSector = 2048;
301 break;
302 case 0x3: /* Mode 2 formless */
303 cbSector = 2336;
304 break;
305 case 0x4: /* Mode 2 form 1 */
306 cbSector = 2048;
307 break;
308 case 0x5: /* Mode 2 form 2 */
309 cbSector = 2324;
310 break;
311 default: /* Reserved */
312 AssertMsgFailed(("Unknown sector type\n"));
313 cbSector = 0; /** @todo we should probably fail the command here already. */
314 }
315
316 if (pbCdb[0] == SCSI_READ_CD)
317 cbXfer = drvHostDvdBE2H_U24(pbCdb + 6) * cbSector;
318 else /* SCSI_READ_MSF */
319 {
320 cSectors = drvHostDvdMSF2LBA(pbCdb + 6) - drvHostDvdMSF2LBA(pbCdb + 3);
321 if (cSectors > 32)
322 cSectors = 32; /* Limit transfer size to 64~74K. Safety first. In any case this can only harm software doing CDDA extraction. */
323 cbXfer = cSectors * cbSector;
324 }
325 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
326 fPassthrough = true;
327 break;
328 }
329 case SCSI_READ_DISC_INFORMATION:
330 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
331 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
332 fPassthrough = true;
333 break;
334 case SCSI_READ_DVD_STRUCTURE:
335 cbXfer = drvHostDvdBE2H_U16(pbCdb + 8);
336 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
337 fPassthrough = true;
338 break;
339 case SCSI_READ_FORMAT_CAPACITIES:
340 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
341 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
342 fPassthrough = true;
343 break;
344 case SCSI_READ_SUBCHANNEL:
345 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
346 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
347 fPassthrough = true;
348 break;
349 case SCSI_READ_TOC_PMA_ATIP:
350 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
351 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
352 fPassthrough = true;
353 break;
354 case SCSI_READ_TRACK_INFORMATION:
355 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
356 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
357 fPassthrough = true;
358 break;
359 case SCSI_REPORT_KEY:
360 cbXfer = drvHostDvdBE2H_U16(pbCdb + 8);
361 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
362 fPassthrough = true;
363 break;
364 case SCSI_REQUEST_SENSE:
365 cbXfer = pbCdb[4];
366 if ((pThis->abATAPISense[2] & 0x0f) != SCSI_SENSE_NONE)
367 {
368 /* Copy sense data over. */
369 void *pvBuf = NULL;
370 int rc = drvHostBaseBufferRetain(&pThis->Core, pReq, cbBuf, false /*fWrite*/, &pvBuf);
371 if (RT_SUCCESS(rc))
372 {
373 memcpy(pvBuf, &pThis->abATAPISense[0], RT_MIN(sizeof(pThis->abATAPISense), cbBuf));
374 rc = drvHostBaseBufferRelease(&pThis->Core, pReq, cbBuf, false /* fWrite */, pvBuf);
375 AssertRC(rc);
376 drvHostDvdCmdOK(pThis);
377 *pu8ScsiSts = SCSI_STATUS_OK;
378 }
379 break;
380 }
381 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
382 fPassthrough = true;
383 break;
384 case SCSI_SEND_CUE_SHEET:
385 cbXfer = drvHostDvdBE2H_U24(pbCdb + 6);
386 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
387 fPassthrough = true;
388 break;
389 case SCSI_SEND_DVD_STRUCTURE:
390 cbXfer = drvHostDvdBE2H_U16(pbCdb + 8);
391 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
392 fPassthrough = true;
393 break;
394 case SCSI_SEND_EVENT:
395 cbXfer = drvHostDvdBE2H_U16(pbCdb + 8);
396 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
397 fPassthrough = true;
398 break;
399 case SCSI_SEND_KEY:
400 cbXfer = drvHostDvdBE2H_U16(pbCdb + 8);
401 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
402 fPassthrough = true;
403 break;
404 case SCSI_SEND_OPC_INFORMATION:
405 cbXfer = drvHostDvdBE2H_U16(pbCdb + 7);
406 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
407 fPassthrough = true;
408 break;
409 case SCSI_SET_STREAMING:
410 cbXfer = drvHostDvdBE2H_U16(pbCdb + 9);
411 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
412 fPassthrough = true;
413 break;
414 case SCSI_WRITE_10:
415 case SCSI_WRITE_AND_VERIFY_10:
416 uLba = drvHostDvdBE2H_U32(pbCdb + 2);
417 cSectors = drvHostDvdBE2H_U16(pbCdb + 7);
418 if (pThis->pTrackList)
419 cbSector = ATAPIPassthroughTrackListGetSectorSizeFromLba(pThis->pTrackList, uLba);
420 else
421 cbSector = 2048;
422 cbXfer = cSectors * cbSector;
423 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
424 fPassthrough = true;
425 break;
426 case SCSI_WRITE_12:
427 uLba = drvHostDvdBE2H_U32(pbCdb + 2);
428 cSectors = drvHostDvdBE2H_U32(pbCdb + 6);
429 if (pThis->pTrackList)
430 cbSector = ATAPIPassthroughTrackListGetSectorSizeFromLba(pThis->pTrackList, uLba);
431 else
432 cbSector = 2048;
433 cbXfer = cSectors * cbSector;
434 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
435 fPassthrough = true;
436 break;
437 case SCSI_WRITE_BUFFER:
438 switch (pbCdb[1] & 0x1f)
439 {
440 case 0x04: /* download microcode */
441 case 0x05: /* download microcode and save */
442 case 0x06: /* download microcode with offsets */
443 case 0x07: /* download microcode with offsets and save */
444 case 0x0e: /* download microcode with offsets and defer activation */
445 case 0x0f: /* activate deferred microcode */
446 LogRel(("HostDVD#%u: CD-ROM passthrough command attempted to update firmware, blocked\n", pThis->Core.pDrvIns->iInstance));
447 *pu8ScsiSts = drvHostDvdCmdErrorSimple(pThis, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET);
448 break;
449 default:
450 cbXfer = drvHostDvdBE2H_U16(pbCdb + 6);
451 enmTxDir = PDMMEDIATXDIR_TO_DEVICE;
452 fPassthrough = true;
453 break;
454 }
455 break;
456 case SCSI_REPORT_LUNS: /* Not part of MMC-3, but used by Windows. */
457 cbXfer = drvHostDvdBE2H_U32(pbCdb + 6);
458 enmTxDir = PDMMEDIATXDIR_FROM_DEVICE;
459 fPassthrough = true;
460 break;
461 case SCSI_REZERO_UNIT:
462 /* Obsolete command used by cdrecord. What else would one expect?
463 * This command is not sent to the drive, it is handled internally,
464 * as the Linux kernel doesn't like it (message "scsi: unknown
465 * opcode 0x01" in syslog) and replies with a sense code of 0,
466 * which sends cdrecord to an endless loop. */
467 *pu8ScsiSts = drvHostDvdCmdErrorSimple(pThis, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE);
468 break;
469 default:
470 LogRel(("HostDVD#%u: Passthrough unimplemented for command %#x\n", pThis->Core.pDrvIns->iInstance, pbCdb[0]));
471 *pu8ScsiSts = drvHostDvdCmdErrorSimple(pThis, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE);
472 break;
473 }
474
475 if (fPassthrough)
476 {
477 *penmTxDir = enmTxDir;
478 *pcbXfer = cbXfer;
479 *pcbSector = cbSector;
480 }
481
482 return fPassthrough;
483}
484
485/**
486 * Locks or unlocks the drive.
487 *
488 * @returns VBox status code.
489 * @param pThis The instance data.
490 * @param fLock True if the request is to lock the drive, false if to unlock.
491 */
492static DECLCALLBACK(int) drvHostDvdDoLock(PDRVHOSTBASE pThis, bool fLock)
493{
494 int rc = drvHostBaseDoLockOs(pThis, fLock);
495
496 LogFlow(("drvHostDvdDoLock(, fLock=%RTbool): returns %Rrc\n", fLock, rc));
497 return rc;
498}
499
500
501/** @interface_method_impl{PDMIMEDIA,pfnSendCmd} */
502static DECLCALLBACK(int) drvHostDvdSendCmd(PPDMIMEDIA pInterface, const uint8_t *pbCmd,
503 PDMMEDIATXDIR enmTxDir, void *pvBuf, uint32_t *pcbBuf,
504 uint8_t *pabSense, size_t cbSense, uint32_t cTimeoutMillies)
505{
506 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
507 int rc;
508 LogFlow(("%s: cmd[0]=%#04x txdir=%d pcbBuf=%d timeout=%d\n", __FUNCTION__, pbCmd[0], enmTxDir, *pcbBuf, cTimeoutMillies));
509
510 RTCritSectEnter(&pThis->CritSect);
511 /*
512 * Pass the request on to the internal scsi command interface.
513 * The command seems to be 12 bytes long, the docs a bit copy&pasty on the command length point...
514 */
515 if (enmTxDir == PDMMEDIATXDIR_FROM_DEVICE)
516 memset(pvBuf, '\0', *pcbBuf); /* we got read size, but zero it anyway. */
517 rc = drvHostBaseScsiCmdOs(pThis, pbCmd, 12, enmTxDir, pvBuf, pcbBuf, pabSense, cbSense, cTimeoutMillies);
518 if (rc == VERR_UNRESOLVED_ERROR)
519 /* sense information set */
520 rc = VERR_DEV_IO_ERROR;
521
522 if (pbCmd[0] == SCSI_GET_EVENT_STATUS_NOTIFICATION)
523 {
524 uint8_t *pbBuf = (uint8_t*)pvBuf;
525 Log2(("Event Status Notification class=%#02x supported classes=%#02x\n", pbBuf[2], pbBuf[3]));
526 if (RT_BE2H_U16(*(uint16_t*)pbBuf) >= 6)
527 Log2((" event %#02x %#02x %#02x %#02x\n", pbBuf[4], pbBuf[5], pbBuf[6], pbBuf[7]));
528 }
529 RTCritSectLeave(&pThis->CritSect);
530
531 LogFlow(("%s: rc=%Rrc\n", __FUNCTION__, rc));
532 return rc;
533}
534
535
536/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqSendScsiCmd} */
537static DECLCALLBACK(int) drvHostDvdIoReqSendScsiCmd(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, uint32_t uLun,
538 const uint8_t *pbCdb, size_t cbCdb, PDMMEDIAEXIOREQSCSITXDIR enmTxDir,
539 size_t cbBuf, uint8_t *pabSense, size_t cbSense, uint8_t *pu8ScsiSts,
540 uint32_t cTimeoutMillies)
541{
542 RT_NOREF3(uLun, cTimeoutMillies, enmTxDir);
543
544 PDRVHOSTDVD pThis = RT_FROM_MEMBER(pInterface, DRVHOSTDVD, Core.IMediaEx);
545 PDRVHOSTBASEREQ pReq = (PDRVHOSTBASEREQ)hIoReq;
546 int rc = VINF_SUCCESS;
547
548 LogFlow(("%s: pbCdb[0]=%#04x{%s} enmTxDir=%d cbBuf=%zu timeout=%u\n",
549 __FUNCTION__, pbCdb[0], SCSICmdText(pbCdb[0]), enmTxDir, cbBuf, cTimeoutMillies));
550
551 RTCritSectEnter(&pThis->Core.CritSect);
552
553 /*
554 * Parse the command first to fend off any illegal or dangeroups commands we don't want the guest
555 * to execute on the host drive.
556 */
557 PDMMEDIATXDIR enmXferDir = PDMMEDIATXDIR_NONE;
558 size_t cbXfer = 0;
559 size_t cbSector = 0;
560 bool fPassthrough = drvHostDvdParseCdb(pThis, pReq, pbCdb, cbCdb, cbBuf,
561 &enmXferDir, &cbXfer, &cbSector, pu8ScsiSts);
562 if (fPassthrough)
563 {
564 void *pvBuf = NULL;
565 size_t cbXferCur = cbXfer;
566
567 pReq->cbReq = cbXfer;
568 pReq->cbResidual = cbXfer;
569
570 if (cbXfer)
571 rc = drvHostBaseBufferRetain(&pThis->Core, pReq, cbXfer, enmXferDir == PDMMEDIATXDIR_TO_DEVICE, &pvBuf);
572
573 if (cbXfer > SCSI_MAX_BUFFER_SIZE)
574 {
575 /* Linux accepts commands with up to 100KB of data, but expects
576 * us to handle commands with up to 128KB of data. The usual
577 * imbalance of powers. */
578 uint8_t aATAPICmd[ATAPI_PACKET_SIZE];
579 uint32_t iATAPILBA, cSectors;
580 uint8_t *pbBuf = (uint8_t *)pvBuf;
581
582 switch (pbCdb[0])
583 {
584 case SCSI_READ_10:
585 case SCSI_WRITE_10:
586 case SCSI_WRITE_AND_VERIFY_10:
587 iATAPILBA = drvHostDvdBE2H_U32(pbCdb + 2);
588 cSectors = drvHostDvdBE2H_U16(pbCdb + 7);
589 break;
590 case SCSI_READ_12:
591 case SCSI_WRITE_12:
592 iATAPILBA = drvHostDvdBE2H_U32(pbCdb + 2);
593 cSectors = drvHostDvdBE2H_U32(pbCdb + 6);
594 break;
595 case SCSI_READ_CD:
596 iATAPILBA = drvHostDvdBE2H_U32(pbCdb + 2);
597 cSectors = drvHostDvdBE2H_U24(pbCdb + 6);
598 break;
599 case SCSI_READ_CD_MSF:
600 iATAPILBA = drvHostDvdMSF2LBA(pbCdb + 3);
601 cSectors = drvHostDvdMSF2LBA(pbCdb + 6) - iATAPILBA;
602 break;
603 default:
604 AssertMsgFailed(("Don't know how to split command %#04x\n", pbCdb[0]));
605 LogRelMax(10, ("HostDVD#%u: CD-ROM passthrough split error\n", pThis->Core.pDrvIns->iInstance));
606 *pu8ScsiSts = drvHostDvdCmdErrorSimple(pThis, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE);
607 rc = drvHostBaseBufferRelease(&pThis->Core, pReq, cbBuf, enmXferDir == PDMMEDIATXDIR_TO_DEVICE, pvBuf);
608 RTCritSectLeave(&pThis->Core.CritSect);
609 return VINF_SUCCESS;
610 }
611 memcpy(aATAPICmd, pbCdb, RT_MIN(cbCdb, ATAPI_PACKET_SIZE));
612 uint32_t cReqSectors = 0;
613 for (uint32_t i = cSectors; i > 0; i -= cReqSectors)
614 {
615 if (i * cbSector > SCSI_MAX_BUFFER_SIZE)
616 cReqSectors = SCSI_MAX_BUFFER_SIZE / (uint32_t)cbSector;
617 else
618 cReqSectors = i;
619 uint32_t cbCurrTX = (uint32_t)cbSector * cReqSectors;
620 switch (pbCdb[0])
621 {
622 case SCSI_READ_10:
623 case SCSI_WRITE_10:
624 case SCSI_WRITE_AND_VERIFY_10:
625 drvHostDvdH2BE_U32(aATAPICmd + 2, iATAPILBA);
626 drvHostDvdH2BE_U16(aATAPICmd + 7, cReqSectors);
627 break;
628 case SCSI_READ_12:
629 case SCSI_WRITE_12:
630 drvHostDvdH2BE_U32(aATAPICmd + 2, iATAPILBA);
631 drvHostDvdH2BE_U32(aATAPICmd + 6, cReqSectors);
632 break;
633 case SCSI_READ_CD:
634 drvHostDvdH2BE_U32(aATAPICmd + 2, iATAPILBA);
635 drvHostDvdH2BE_U24(aATAPICmd + 6, cReqSectors);
636 break;
637 case SCSI_READ_CD_MSF:
638 drvHostDvdLBA2MSF(aATAPICmd + 3, iATAPILBA);
639 drvHostDvdLBA2MSF(aATAPICmd + 6, iATAPILBA + cReqSectors);
640 break;
641 }
642 rc = drvHostBaseScsiCmdOs(&pThis->Core, aATAPICmd, sizeof(aATAPICmd),
643 enmXferDir, pbBuf, &cbCurrTX,
644 &pThis->abATAPISense[0], sizeof(pThis->abATAPISense),
645 cTimeoutMillies /**< @todo timeout */);
646 if (rc != VINF_SUCCESS)
647 break;
648
649 pReq->cbResidual -= cbCurrTX;
650 iATAPILBA += cReqSectors;
651 pbBuf += cbSector * cReqSectors;
652 }
653 }
654 else
655 {
656 uint32_t cbXferTmp = (uint32_t)cbXferCur;
657 rc = drvHostBaseScsiCmdOs(&pThis->Core, pbCdb, cbCdb, enmXferDir, pvBuf, &cbXferTmp,
658 &pThis->abATAPISense[0], sizeof(pThis->abATAPISense), cTimeoutMillies);
659 if (RT_SUCCESS(rc))
660 pReq->cbResidual -= cbXferTmp;
661 }
662
663 if (RT_SUCCESS(rc))
664 {
665 /* Do post processing for certain commands. */
666 switch (pbCdb[0])
667 {
668 case SCSI_SEND_CUE_SHEET:
669 case SCSI_READ_TOC_PMA_ATIP:
670 {
671 if (!pThis->pTrackList)
672 rc = ATAPIPassthroughTrackListCreateEmpty(&pThis->pTrackList);
673
674 if (RT_SUCCESS(rc))
675 rc = ATAPIPassthroughTrackListUpdate(pThis->pTrackList, pbCdb, pvBuf);
676
677 if (RT_FAILURE(rc))
678 LogRelMax(10, ("HostDVD#%u: Error (%Rrc) while updating the tracklist during %s, burning the disc might fail\n",
679 pThis->Core.pDrvIns->iInstance, rc,
680 pbCdb[0] == SCSI_SEND_CUE_SHEET ? "SEND CUE SHEET" : "READ TOC/PMA/ATIP"));
681 break;
682 }
683 case SCSI_SYNCHRONIZE_CACHE:
684 {
685 if (pThis->pTrackList)
686 ATAPIPassthroughTrackListClear(pThis->pTrackList);
687 break;
688 }
689 }
690
691 if (enmXferDir == PDMMEDIATXDIR_FROM_DEVICE)
692 {
693 Assert(cbXferCur <= cbXfer);
694
695 if (pbCdb[0] == SCSI_INQUIRY)
696 {
697 /* Make sure that the real drive cannot be identified.
698 * Motivation: changing the VM configuration should be as
699 * invisible as possible to the guest. */
700 if (cbXferCur >= 8 + 8)
701 drvHostDvdSCSIPadStr((uint8_t *)pvBuf + 8, "VBOX", 8);
702 if (cbXferCur >= 16 + 16)
703 drvHostDvdSCSIPadStr((uint8_t *)pvBuf + 16, "CD-ROM", 16);
704 if (cbXferCur >= 32 + 4)
705 drvHostDvdSCSIPadStr((uint8_t *)pvBuf + 32, "1.0", 4);
706 }
707
708 if (cbXferCur)
709 Log3(("ATAPI PT data read (%d): %.*Rhxs\n", cbXferCur, cbXferCur, (uint8_t *)pvBuf));
710 }
711
712 *pu8ScsiSts = drvHostDvdCmdOK(pThis);
713 }
714 else
715 {
716 do
717 {
718 /* don't log superfluous errors */
719 if ( rc == VERR_DEV_IO_ERROR
720 && ( pbCdb[0] == SCSI_TEST_UNIT_READY
721 || pbCdb[0] == SCSI_READ_CAPACITY
722 || pbCdb[0] == SCSI_READ_DVD_STRUCTURE
723 || pbCdb[0] == SCSI_READ_TOC_PMA_ATIP))
724 break;
725 LogRelMax(10, ("HostDVD#%u: CD-ROM passthrough cmd=%#04x sense=%d ASC=%#02x ASCQ=%#02x %Rrc\n",
726 pThis->Core.pDrvIns->iInstance, pbCdb[0], pThis->abATAPISense[2] & 0x0f,
727 pThis->abATAPISense[12], pThis->abATAPISense[13], rc));
728 } while (0);
729 *pu8ScsiSts = SCSI_STATUS_CHECK_CONDITION;
730 rc = VINF_SUCCESS;
731 }
732
733 if (cbXfer)
734 rc = drvHostBaseBufferRelease(&pThis->Core, pReq, cbXfer, enmXferDir == PDMMEDIATXDIR_TO_DEVICE, pvBuf);
735 }
736
737 /*
738 * We handled the command, check the status code and copy over the sense data if
739 * it is CHECK CONDITION.
740 */
741 if ( *pu8ScsiSts == SCSI_STATUS_CHECK_CONDITION
742 && VALID_PTR(pabSense)
743 && cbSense > 0)
744 memcpy(pabSense, &pThis->abATAPISense[0], RT_MIN(cbSense, sizeof(pThis->abATAPISense)));
745
746 RTCritSectLeave(&pThis->Core.CritSect);
747
748 LogFlow(("%s: rc=%Rrc\n", __FUNCTION__, rc));
749 return rc;
750}
751
752
753/* -=-=-=-=- driver interface -=-=-=-=- */
754
755
756/** @interface_method_impl{PDMDRVREG,pfnDestruct} */
757static DECLCALLBACK(void) drvHostDvdDestruct(PPDMDRVINS pDrvIns)
758{
759 PDRVHOSTDVD pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTDVD);
760
761 if (pThis->pTrackList)
762 {
763 ATAPIPassthroughTrackListDestroy(pThis->pTrackList);
764 pThis->pTrackList = NULL;
765 }
766
767 DRVHostBaseDestruct(pDrvIns);
768}
769
770/**
771 * Construct a host dvd drive driver instance.
772 *
773 * @copydoc FNPDMDRVCONSTRUCT
774 */
775static DECLCALLBACK(int) drvHostDvdConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
776{
777 RT_NOREF(fFlags);
778 PDRVHOSTDVD pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTDVD);
779 LogFlow(("drvHostDvdConstruct: iInstance=%d\n", pDrvIns->iInstance));
780
781 bool fPassthrough;
782 int rc = CFGMR3QueryBool(pCfg, "Passthrough", &fPassthrough);
783 if (RT_SUCCESS(rc) && fPassthrough)
784 {
785 pThis->Core.IMedia.pfnSendCmd = drvHostDvdSendCmd;
786 pThis->Core.IMediaEx.pfnIoReqSendScsiCmd = drvHostDvdIoReqSendScsiCmd;
787 /* Passthrough requires opening the device in R/W mode. */
788 pThis->Core.fReadOnlyConfig = false;
789 }
790
791 pThis->Core.pfnDoLock = drvHostDvdDoLock;
792
793 /*
794 * Init instance data.
795 */
796 rc = DRVHostBaseInit(pDrvIns, pCfg, "Path\0Interval\0Locked\0BIOSVisible\0AttachFailError\0Passthrough\0",
797 PDMMEDIATYPE_DVD);
798 LogFlow(("drvHostDvdConstruct: returns %Rrc\n", rc));
799 return rc;
800}
801
802
803/**
804 * Block driver registration record.
805 */
806const PDMDRVREG g_DrvHostDVD =
807{
808 /* u32Version */
809 PDM_DRVREG_VERSION,
810 /* szName */
811 "HostDVD",
812 /* szRCMod */
813 "",
814 /* szR0Mod */
815 "",
816 /* pszDescription */
817 "Host DVD Block Driver.",
818 /* fFlags */
819 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
820 /* fClass. */
821 PDM_DRVREG_CLASS_BLOCK,
822 /* cMaxInstances */
823 ~0U,
824 /* cbInstance */
825 sizeof(DRVHOSTDVD),
826 /* pfnConstruct */
827 drvHostDvdConstruct,
828 /* pfnDestruct */
829 drvHostDvdDestruct,
830 /* pfnRelocate */
831 NULL,
832 /* pfnIOCtl */
833 NULL,
834 /* pfnPowerOn */
835 NULL,
836 /* pfnReset */
837 NULL,
838 /* pfnSuspend */
839 NULL,
840 /* pfnResume */
841 NULL,
842 /* pfnAttach */
843 NULL,
844 /* pfnDetach */
845 NULL,
846 /* pfnPowerOff */
847 NULL,
848 /* pfnSoftReset */
849 NULL,
850 /* u32EndVersion */
851 PDM_DRVREG_VERSION
852};
853
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