VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/VSCSI/VSCSILunMmc.cpp@ 67494

Last change on this file since 67494 was 66955, checked in by vboxsync, 8 years ago

Devices/Storage/DrvSCSI,VSCSI: Support CD/DVD images with multiple tracks like for DevATA to bring the AHCI and SCSI controllers up to the same feature level

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 68.7 KB
Line 
1/* $Id: VSCSILunMmc.cpp 66955 2017-05-18 15:57:16Z vboxsync $ */
2/** @file
3 * Virtual SCSI driver: MMC LUN implementation (CD/DVD-ROM)
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_VSCSI
23#include <VBox/log.h>
24#include <VBox/err.h>
25#include <VBox/types.h>
26#include <VBox/vscsi.h>
27#include <iprt/asm.h>
28#include <iprt/assert.h>
29#include <iprt/mem.h>
30#include <iprt/string.h>
31
32#include "VSCSIInternal.h"
33
34/*********************************************************************************************************************************
35* Structures and Typedefs *
36*********************************************************************************************************************************/
37
38/**
39 * Different event status types.
40 */
41typedef enum MMCEVENTSTATUSTYPE
42{
43 /** Medium event status not changed. */
44 MMCEVENTSTATUSTYPE_UNCHANGED = 0,
45 /** New medium inserted. */
46 MMCEVENTSTATUSTYPE_MEDIA_NEW,
47 /** Medium removed. */
48 MMCEVENTSTATUSTYPE_MEDIA_REMOVED,
49 /** Medium was removed + new medium was inserted. */
50 MMCEVENTSTATUSTYPE_MEDIA_CHANGED,
51 /** Medium eject requested (eject button pressed). */
52 MMCEVENTSTATUSTYPE_MEDIA_EJECT_REQUESTED,
53 /** 32bit hack. */
54 MMCEVENTSTATUSTYPE_32BIT_HACK = 0x7fffffff
55} MMCEVENTSTATUSTYPE;
56
57/** @name Media track types.
58 * @{ */
59/** Unknown media type. */
60#define MMC_MEDIA_TYPE_UNKNOWN 0
61/** Door closed, no media. */
62#define MMC_MEDIA_TYPE_NO_DISC 0x70
63/** @} */
64
65
66/**
67 * MMC LUN instance
68 */
69typedef struct VSCSILUNMMC
70{
71 /** Core LUN structure */
72 VSCSILUNINT Core;
73 /** Size of the virtual disk. */
74 uint64_t cSectors;
75 /** Medium locked indicator. */
76 bool fLocked;
77 /** Media event status. */
78 volatile MMCEVENTSTATUSTYPE MediaEventStatus;
79 /** Media track type. */
80 volatile uint32_t u32MediaTrackType;
81} VSCSILUNMMC, *PVSCSILUNMMC;
82
83
84/**
85 * Callback to fill a feature for a GET CONFIGURATION request.
86 *
87 * @returns Number of bytes used for this feature in the buffer.
88 * @param pbBuf The buffer to use.
89 * @param cbBuf Size of the buffer.
90 */
91typedef DECLCALLBACK(size_t) FNVSCSILUNMMCFILLFEATURE(uint8_t *pbBuf, size_t cbBuf);
92/** Pointer to a fill feature callback. */
93typedef FNVSCSILUNMMCFILLFEATURE *PFNVSCSILUNMMCFILLFEATURE;
94
95/**
96 * VSCSI MMC feature descriptor.
97 */
98typedef struct VSCSILUNMMCFEATURE
99{
100 /** The feature number. */
101 uint16_t u16Feat;
102 /** The callback to call for this feature. */
103 PFNVSCSILUNMMCFILLFEATURE pfnFeatureFill;
104} VSCSILUNMMCFEATURE;
105/** Pointer to a VSCSI MMC feature descriptor. */
106typedef VSCSILUNMMCFEATURE *PVSCSILUNMMCFEATURE;
107/** Pointer to a const VSCSI MMC feature descriptor. */
108typedef const VSCSILUNMMCFEATURE *PCVSCSILUNMMCFEATURE;
109
110
111
112/*********************************************************************************************************************************
113* Internal Functions *
114*********************************************************************************************************************************/
115RT_C_DECLS_BEGIN
116static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureListProfiles(uint8_t *pbBuf, size_t cbBuf);
117static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureCore(uint8_t *pbBuf, size_t cbBuf);
118static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureMorphing(uint8_t *pbBuf, size_t cbBuf);
119static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureRemovableMedium(uint8_t *pbBuf, size_t cbBuf);
120static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureRandomReadable(uint8_t *pbBuf, size_t cbBuf);
121static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureCDRead(uint8_t *pbBuf, size_t cbBuf);
122static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeaturePowerManagement(uint8_t *pbBuf, size_t cbBuf);
123static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureTimeout(uint8_t *pbBuf, size_t cbBuf);
124RT_C_DECLS_END
125
126/**
127 * List of supported MMC features.
128 */
129static const VSCSILUNMMCFEATURE g_aVScsiMmcFeatures[] =
130{
131 { 0x0000, vscsiLunMmcGetConfigurationFillFeatureListProfiles},
132 { 0x0001, vscsiLunMmcGetConfigurationFillFeatureCore},
133 { 0x0002, vscsiLunMmcGetConfigurationFillFeatureMorphing},
134 { 0x0003, vscsiLunMmcGetConfigurationFillFeatureRemovableMedium},
135 { 0x0010, vscsiLunMmcGetConfigurationFillFeatureRandomReadable},
136 { 0x001e, vscsiLunMmcGetConfigurationFillFeatureCDRead},
137 { 0x0100, vscsiLunMmcGetConfigurationFillFeaturePowerManagement},
138 { 0x0105, vscsiLunMmcGetConfigurationFillFeatureTimeout}
139};
140
141/* Fabricate normal TOC information. */
142static int mmcReadTOCNormal(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq, uint16_t cbMaxTransfer, bool fMSF)
143{
144 uint8_t aReply[2+99*8 + 32]; RT_ZERO(aReply); /* Maximum possible reply plus some safety. */
145 uint8_t *pbBuf = aReply;
146 uint8_t *q;
147 uint8_t iStartTrack;
148 uint32_t cbSize;
149 uint32_t cTracks = vscsiLunMediumGetRegionCount(pVScsiLun);
150
151 iStartTrack = pVScsiReq->pbCDB[6];
152 if (iStartTrack == 0)
153 iStartTrack = 1;
154 if (iStartTrack > cTracks && iStartTrack != 0xaa)
155 return vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
156
157 q = pbBuf + 2;
158 *q++ = iStartTrack; /* first track number */
159 *q++ = cTracks; /* last track number */
160 for (uint32_t iTrack = iStartTrack; iTrack <= cTracks; iTrack++)
161 {
162 uint64_t uLbaStart = 0;
163 VDREGIONDATAFORM enmDataForm = VDREGIONDATAFORM_MODE1_2048;
164
165 int rc = vscsiLunMediumQueryRegionProperties(pVScsiLun, iTrack - 1, &uLbaStart,
166 NULL, NULL, &enmDataForm);
167 AssertRC(rc);
168
169 *q++ = 0; /* reserved */
170
171 if (enmDataForm == VDREGIONDATAFORM_CDDA)
172 *q++ = 0x10; /* ADR, control */
173 else
174 *q++ = 0x14; /* ADR, control */
175
176 *q++ = (uint8_t)iTrack; /* track number */
177 *q++ = 0; /* reserved */
178 if (fMSF)
179 {
180 *q++ = 0; /* reserved */
181 scsiLBA2MSF(q, (uint32_t)uLbaStart);
182 q += 3;
183 }
184 else
185 {
186 /* sector 0 */
187 scsiH2BE_U32(q, (uint32_t)uLbaStart);
188 q += 4;
189 }
190 }
191 /* lead out track */
192 *q++ = 0; /* reserved */
193 *q++ = 0x14; /* ADR, control */
194 *q++ = 0xaa; /* track number */
195 *q++ = 0; /* reserved */
196
197 /* Query start and length of last track to get the start of the lead out track. */
198 uint64_t uLbaStart = 0;
199 uint64_t cBlocks = 0;
200
201 int rc = vscsiLunMediumQueryRegionProperties(pVScsiLun, cTracks - 1, &uLbaStart,
202 &cBlocks, NULL, NULL);
203 AssertRC(rc);
204
205 uLbaStart += cBlocks;
206 if (fMSF)
207 {
208 *q++ = 0; /* reserved */
209 scsiLBA2MSF(q, (uint32_t)uLbaStart);
210 q += 3;
211 }
212 else
213 {
214 scsiH2BE_U32(q, (uint32_t)uLbaStart);
215 q += 4;
216 }
217 cbSize = q - pbBuf;
218 Assert(cbSize <= sizeof(aReply));
219 scsiH2BE_U16(pbBuf, cbSize - 2);
220 if (cbSize < cbMaxTransfer)
221 cbMaxTransfer = cbSize;
222
223 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, cbMaxTransfer);
224 return vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
225}
226
227/* Fabricate session information. */
228static int mmcReadTOCMulti(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq, uint16_t cbMaxTransfer, bool fMSF)
229{
230 RT_NOREF1(cbMaxTransfer);
231 uint8_t aReply[32];
232 uint8_t *pbBuf = aReply;
233
234 /* multi session: only a single session defined */
235 memset(pbBuf, 0, 12);
236 pbBuf[1] = 0x0a;
237 pbBuf[2] = 0x01; /* first complete session number */
238 pbBuf[3] = 0x01; /* last complete session number */
239
240 VDREGIONDATAFORM enmDataForm = VDREGIONDATAFORM_MODE1_2048;
241 int rc = vscsiLunMediumQueryRegionProperties(pVScsiLun, 0, NULL,
242 NULL, NULL, &enmDataForm);
243 AssertRC(rc);
244
245 if (enmDataForm == VDREGIONDATAFORM_CDDA)
246 pbBuf[5] = 0x10; /* ADR, control */
247 else
248 pbBuf[5] = 0x14; /* ADR, control */
249
250 pbBuf[6] = 1; /* first track in last complete session */
251
252 if (fMSF)
253 {
254 pbBuf[8] = 0; /* reserved */
255 scsiLBA2MSF(pbBuf + 8, 0);
256 }
257 else
258 {
259 /* sector 0 */
260 scsiH2BE_U32(pbBuf + 8, 0);
261 }
262
263 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, 12);
264
265 return vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
266}
267
268/**
269 * Create raw TOC data information.
270 *
271 * @returns SCSI status code.
272 * @param pVScsiLun The LUN instance.
273 * @param pVScsiReq The VSCSI request.
274 * @param cbMaxTransfer The maximum transfer size.
275 * @param fMSF Flag whether to use MSF format to encode sector numbers.
276 */
277static int mmcReadTOCRaw(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq, uint16_t cbMaxTransfer, bool fMSF)
278{
279 PVSCSILUNMMC pVScsiLunMmc = (PVSCSILUNMMC)pVScsiLun;
280 uint8_t aReply[50]; /* Counted a maximum of 45 bytes but better be on the safe side. */
281 uint32_t cbSize;
282 uint8_t *pbBuf = &aReply[0] + 2;
283
284 *pbBuf++ = 1; /* first session */
285 *pbBuf++ = 1; /* last session */
286
287 *pbBuf++ = 1; /* session number */
288 *pbBuf++ = 0x14; /* data track */
289 *pbBuf++ = 0; /* track number */
290 *pbBuf++ = 0xa0; /* first track in program area */
291 *pbBuf++ = 0; /* min */
292 *pbBuf++ = 0; /* sec */
293 *pbBuf++ = 0; /* frame */
294 *pbBuf++ = 0;
295 *pbBuf++ = 1; /* first track */
296 *pbBuf++ = 0x00; /* disk type CD-DA or CD data */
297 *pbBuf++ = 0;
298
299 *pbBuf++ = 1; /* session number */
300 *pbBuf++ = 0x14; /* data track */
301 *pbBuf++ = 0; /* track number */
302 *pbBuf++ = 0xa1; /* last track in program area */
303 *pbBuf++ = 0; /* min */
304 *pbBuf++ = 0; /* sec */
305 *pbBuf++ = 0; /* frame */
306 *pbBuf++ = 0;
307 *pbBuf++ = 1; /* last track */
308 *pbBuf++ = 0;
309 *pbBuf++ = 0;
310
311 *pbBuf++ = 1; /* session number */
312 *pbBuf++ = 0x14; /* data track */
313 *pbBuf++ = 0; /* track number */
314 *pbBuf++ = 0xa2; /* lead-out */
315 *pbBuf++ = 0; /* min */
316 *pbBuf++ = 0; /* sec */
317 *pbBuf++ = 0; /* frame */
318 if (fMSF)
319 {
320 *pbBuf++ = 0; /* reserved */
321 scsiLBA2MSF(pbBuf, pVScsiLunMmc->cSectors);
322 pbBuf += 3;
323 }
324 else
325 {
326 scsiH2BE_U32(pbBuf, pVScsiLunMmc->cSectors);
327 pbBuf += 4;
328 }
329
330 *pbBuf++ = 1; /* session number */
331 *pbBuf++ = 0x14; /* ADR, control */
332 *pbBuf++ = 0; /* track number */
333 *pbBuf++ = 1; /* point */
334 *pbBuf++ = 0; /* min */
335 *pbBuf++ = 0; /* sec */
336 *pbBuf++ = 0; /* frame */
337 if (fMSF)
338 {
339 *pbBuf++ = 0; /* reserved */
340 scsiLBA2MSF(pbBuf, 0);
341 pbBuf += 3;
342 }
343 else
344 {
345 /* sector 0 */
346 scsiH2BE_U32(pbBuf, 0);
347 pbBuf += 4;
348 }
349
350 cbSize = pbBuf - aReply;
351 scsiH2BE_U16(&aReply[0], cbSize - 2);
352
353 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, RT_MIN(cbMaxTransfer, cbSize));
354 return vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
355}
356
357static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureListProfiles(uint8_t *pbBuf, size_t cbBuf)
358{
359 if (cbBuf < 3*4)
360 return 0;
361
362 scsiH2BE_U16(pbBuf, 0x0); /* feature 0: list of profiles supported */
363 pbBuf[2] = (0 << 2) | (1 << 1) | (1 << 0); /* version 0, persistent, current */
364 pbBuf[3] = 8; /* additional bytes for profiles */
365 /* The MMC-3 spec says that DVD-ROM read capability should be reported
366 * before CD-ROM read capability. */
367 scsiH2BE_U16(pbBuf + 4, 0x10); /* profile: read-only DVD */
368 pbBuf[6] = (0 << 0); /* NOT current profile */
369 scsiH2BE_U16(pbBuf + 8, 0x08); /* profile: read only CD */
370 pbBuf[10] = (1 << 0); /* current profile */
371
372 return 3*4; /* Header + 2 profiles entries */
373}
374
375static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureCore(uint8_t *pbBuf, size_t cbBuf)
376{
377 if (cbBuf < 12)
378 return 0;
379
380 scsiH2BE_U16(pbBuf, 0x1); /* feature 0001h: Core Feature */
381 pbBuf[2] = (0x2 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
382 pbBuf[3] = 8; /* Additional length */
383 scsiH2BE_U16(pbBuf + 4, 0x00000002); /* Physical interface ATAPI. */
384 pbBuf[8] = RT_BIT(0); /* DBE */
385 /* Rest is reserved. */
386
387 return 12;
388}
389
390static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureMorphing(uint8_t *pbBuf, size_t cbBuf)
391{
392 if (cbBuf < 8)
393 return 0;
394
395 scsiH2BE_U16(pbBuf, 0x2); /* feature 0002h: Morphing Feature */
396 pbBuf[2] = (0x1 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
397 pbBuf[3] = 4; /* Additional length */
398 pbBuf[4] = RT_BIT(1) | 0x0; /* OCEvent | !ASYNC */
399 /* Rest is reserved. */
400
401 return 8;
402}
403
404static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureRemovableMedium(uint8_t *pbBuf, size_t cbBuf)
405{
406 if (cbBuf < 8)
407 return 0;
408
409 scsiH2BE_U16(pbBuf, 0x3); /* feature 0003h: Removable Medium Feature */
410 pbBuf[2] = (0x2 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
411 pbBuf[3] = 4; /* Additional length */
412 /* Tray type loading | Load | Eject | !Pvnt Jmpr | !DBML | Lock */
413 pbBuf[4] = (0x2 << 5) | RT_BIT(4) | RT_BIT(3) | (0x0 << 2) | (0x0 << 1) | RT_BIT(0);
414 /* Rest is reserved. */
415
416 return 8;
417}
418
419static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureRandomReadable(uint8_t *pbBuf, size_t cbBuf)
420{
421 if (cbBuf < 12)
422 return 0;
423
424 scsiH2BE_U16(pbBuf, 0x10); /* feature 0010h: Random Readable Feature */
425 pbBuf[2] = (0x0 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
426 pbBuf[3] = 8; /* Additional length */
427 scsiH2BE_U32(pbBuf + 4, 2048); /* Logical block size. */
428 scsiH2BE_U16(pbBuf + 8, 0x10); /* Blocking (0x10 for DVD, CD is not defined). */
429 pbBuf[10] = 0; /* PP not present */
430 /* Rest is reserved. */
431
432 return 12;
433}
434
435static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureCDRead(uint8_t *pbBuf, size_t cbBuf)
436{
437 if (cbBuf < 8)
438 return 0;
439
440 scsiH2BE_U16(pbBuf, 0x1e); /* feature 001Eh: CD Read Feature */
441 pbBuf[2] = (0x2 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
442 pbBuf[3] = 0; /* Additional length */
443 pbBuf[4] = (0x0 << 7) | (0x0 << 1) | 0x0; /* !DAP | !C2-Flags | !CD-Text. */
444 /* Rest is reserved. */
445
446 return 8;
447}
448
449static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeaturePowerManagement(uint8_t *pbBuf, size_t cbBuf)
450{
451 if (cbBuf < 4)
452 return 0;
453
454 scsiH2BE_U16(pbBuf, 0x100); /* feature 0100h: Power Management Feature */
455 pbBuf[2] = (0x0 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
456 pbBuf[3] = 0; /* Additional length */
457
458 return 4;
459}
460
461static DECLCALLBACK(size_t) vscsiLunMmcGetConfigurationFillFeatureTimeout(uint8_t *pbBuf, size_t cbBuf)
462{
463 if (cbBuf < 8)
464 return 0;
465
466 scsiH2BE_U16(pbBuf, 0x105); /* feature 0105h: Timeout Feature */
467 pbBuf[2] = (0x0 << 2) | RT_BIT(1) | RT_BIT(0); /* Version | Persistent | Current */
468 pbBuf[3] = 4; /* Additional length */
469 pbBuf[4] = 0x0; /* !Group3 */
470
471 return 8;
472}
473
474/**
475 * Processes the GET CONFIGURATION SCSI request.
476 *
477 * @returns SCSI status code.
478 * @param pVScsiLunMmc The MMC LUN instance.
479 * @param pVScsiReq The VSCSI request.
480 * @param cbMaxTransfer The maximum transfer size.
481 */
482static int vscsiLunMmcGetConfiguration(PVSCSILUNMMC pVScsiLunMmc, PVSCSIREQINT pVScsiReq, size_t cbMaxTransfer)
483{
484 uint8_t aReply[80];
485 uint8_t *pbBuf = &aReply[0];
486 size_t cbBuf = sizeof(aReply);
487 size_t cbCopied = 0;
488 uint16_t u16Sfn = scsiBE2H_U16(&pVScsiReq->pbCDB[2]);
489 uint8_t u8Rt = pVScsiReq->pbCDB[1] & 0x03;
490
491 /* Accept valid request types only. */
492 if (u8Rt == 3)
493 return vscsiLunReqSenseErrorSet(&pVScsiLunMmc->Core, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
494 SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
495
496 /** @todo implement switching between CD-ROM and DVD-ROM profile (the only
497 * way to differentiate them right now is based on the image size). */
498 if (pVScsiLunMmc->cSectors)
499 scsiH2BE_U16(pbBuf + 6, 0x08); /* current profile: read-only CD */
500 else
501 scsiH2BE_U16(pbBuf + 6, 0x00); /* current profile: none -> no media */
502 cbBuf -= 8;
503 pbBuf += 8;
504
505 if (u8Rt == 0x2)
506 {
507 for (uint32_t i = 0; i < RT_ELEMENTS(g_aVScsiMmcFeatures); i++)
508 {
509 if (g_aVScsiMmcFeatures[i].u16Feat == u16Sfn)
510 {
511 cbCopied = g_aVScsiMmcFeatures[i].pfnFeatureFill(pbBuf, cbBuf);
512 cbBuf -= cbCopied;
513 pbBuf += cbCopied;
514 break;
515 }
516 }
517 }
518 else
519 {
520 for (uint32_t i = 0; i < RT_ELEMENTS(g_aVScsiMmcFeatures); i++)
521 {
522 if (g_aVScsiMmcFeatures[i].u16Feat > u16Sfn)
523 {
524 cbCopied = g_aVScsiMmcFeatures[i].pfnFeatureFill(pbBuf, cbBuf);
525 cbBuf -= cbCopied;
526 pbBuf += cbCopied;
527 }
528 }
529 }
530
531 /* Set data length now. */
532 scsiH2BE_U32(&aReply[0], (uint32_t)(sizeof(aReply) - cbBuf));
533
534 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, RT_MIN(cbMaxTransfer, sizeof(aReply) - cbBuf));
535 return vscsiLunReqSenseOkSet(&pVScsiLunMmc->Core, pVScsiReq);
536}
537
538/**
539 * Processes the READ DVD STRUCTURE SCSI request.
540 *
541 * @returns SCSI status code.
542 * @param pVScsiLunMmc The MMC LUN instance.
543 * @param pVScsiReq The VSCSI request.
544 * @param cbMaxTransfer The maximum transfer size.
545 */
546static int vscsiLunMmcReadDvdStructure(PVSCSILUNMMC pVScsiLunMmc, PVSCSIREQINT pVScsiReq, size_t cbMaxTransfer)
547{
548 uint8_t aReply[25]; /* Counted a maximum of 20 bytes but better be on the safe side. */
549
550 RT_ZERO(aReply);
551
552 /* Act according to the indicated format. */
553 switch (pVScsiReq->pbCDB[7])
554 {
555 case 0x00:
556 case 0x01:
557 case 0x02:
558 case 0x03:
559 case 0x04:
560 case 0x05:
561 case 0x06:
562 case 0x07:
563 case 0x08:
564 case 0x09:
565 case 0x0a:
566 case 0x0b:
567 case 0x0c:
568 case 0x0d:
569 case 0x0e:
570 case 0x0f:
571 case 0x10:
572 case 0x11:
573 case 0x30:
574 case 0x31:
575 case 0xff:
576 if (pVScsiReq->pbCDB[1] == 0)
577 {
578 int uASC = SCSI_ASC_NONE;
579
580 switch (pVScsiReq->pbCDB[7])
581 {
582 case 0x0: /* Physical format information */
583 {
584 uint8_t uLayer = pVScsiReq->pbCDB[6];
585 uint64_t cTotalSectors;
586
587 if (uLayer != 0)
588 {
589 uASC = -SCSI_ASC_INV_FIELD_IN_CMD_PACKET;
590 break;
591 }
592
593 cTotalSectors = pVScsiLunMmc->cSectors;
594 cTotalSectors >>= 2;
595 if (cTotalSectors == 0)
596 {
597 uASC = -SCSI_ASC_MEDIUM_NOT_PRESENT;
598 break;
599 }
600
601 aReply[4] = 1; /* DVD-ROM, part version 1 */
602 aReply[5] = 0xf; /* 120mm disc, minimum rate unspecified */
603 aReply[6] = 1; /* one layer, read-only (per MMC-2 spec) */
604 aReply[7] = 0; /* default densities */
605
606 /* FIXME: 0x30000 per spec? */
607 scsiH2BE_U32(&aReply[8], 0); /* start sector */
608 scsiH2BE_U32(&aReply[12], cTotalSectors - 1); /* end sector */
609 scsiH2BE_U32(&aReply[16], cTotalSectors - 1); /* l0 end sector */
610
611 /* Size of buffer, not including 2 byte size field */
612 scsiH2BE_U32(&aReply[0], 2048 + 2);
613
614 /* 2k data + 4 byte header */
615 uASC = (2048 + 4);
616 break;
617 }
618 case 0x01: /* DVD copyright information */
619 aReply[4] = 0; /* no copyright data */
620 aReply[5] = 0; /* no region restrictions */
621
622 /* Size of buffer, not including 2 byte size field */
623 scsiH2BE_U16(&aReply[0], 4 + 2);
624
625 /* 4 byte header + 4 byte data */
626 uASC = (4 + 4);
627 break;
628
629 case 0x03: /* BCA information - invalid field for no BCA info */
630 uASC = -SCSI_ASC_INV_FIELD_IN_CMD_PACKET;
631 break;
632
633 case 0x04: /* DVD disc manufacturing information */
634 /* Size of buffer, not including 2 byte size field */
635 scsiH2BE_U16(&aReply[0], 2048 + 2);
636
637 /* 2k data + 4 byte header */
638 uASC = (2048 + 4);
639 break;
640 case 0xff:
641 /*
642 * This lists all the command capabilities above. Add new ones
643 * in order and update the length and buffer return values.
644 */
645
646 aReply[4] = 0x00; /* Physical format */
647 aReply[5] = 0x40; /* Not writable, is readable */
648 scsiH2BE_U16(&aReply[6], 2048 + 4);
649
650 aReply[8] = 0x01; /* Copyright info */
651 aReply[9] = 0x40; /* Not writable, is readable */
652 scsiH2BE_U16(&aReply[10], 4 + 4);
653
654 aReply[12] = 0x03; /* BCA info */
655 aReply[13] = 0x40; /* Not writable, is readable */
656 scsiH2BE_U16(&aReply[14], 188 + 4);
657
658 aReply[16] = 0x04; /* Manufacturing info */
659 aReply[17] = 0x40; /* Not writable, is readable */
660 scsiH2BE_U16(&aReply[18], 2048 + 4);
661
662 /* Size of buffer, not including 2 byte size field */
663 scsiH2BE_U16(&aReply[0], 16 + 2);
664
665 /* data written + 4 byte header */
666 uASC = (16 + 4);
667 break;
668 default: /** @todo formats beyond DVD-ROM requires */
669 uASC = -SCSI_ASC_INV_FIELD_IN_CMD_PACKET;
670 }
671
672 if (uASC < 0)
673 return vscsiLunReqSenseErrorSet(&pVScsiLunMmc->Core, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
674 -uASC, 0x00);
675 break;
676 }
677 /** @todo BD support, fall through for now */
678 /* fall thru */
679
680 /* Generic disk structures */
681 case 0x80: /** @todo AACS volume identifier */
682 case 0x81: /** @todo AACS media serial number */
683 case 0x82: /** @todo AACS media identifier */
684 case 0x83: /** @todo AACS media key block */
685 case 0x90: /** @todo List of recognized format layers */
686 case 0xc0: /** @todo Write protection status */
687 default:
688 return vscsiLunReqSenseErrorSet(&pVScsiLunMmc->Core, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
689 SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
690 }
691
692 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, RT_MIN(cbMaxTransfer, sizeof(aReply)));
693 return vscsiLunReqSenseOkSet(&pVScsiLunMmc->Core, pVScsiReq);
694}
695
696/**
697 * Processes the MODE SENSE 10 SCSI request.
698 *
699 * @returns SCSI status code.
700 * @param pVScsiLunMmc The MMC LUN instance.
701 * @param pVScsiReq The VSCSI request.
702 * @param cbMaxTransfer The maximum transfer size.
703 */
704static int vscsiLunMmcModeSense10(PVSCSILUNMMC pVScsiLunMmc, PVSCSIREQINT pVScsiReq, size_t cbMaxTransfer)
705{
706 int rcReq;
707 uint8_t uPageControl = pVScsiReq->pbCDB[2] >> 6;
708 uint8_t uPageCode = pVScsiReq->pbCDB[2] & 0x3f;
709
710 switch (uPageControl)
711 {
712 case SCSI_PAGECONTROL_CURRENT:
713 switch (uPageCode)
714 {
715 case SCSI_MODEPAGE_ERROR_RECOVERY:
716 {
717 uint8_t aReply[16];
718
719 scsiH2BE_U16(&aReply[0], 16 + 6);
720 aReply[2] = (uint8_t)pVScsiLunMmc->u32MediaTrackType;
721 aReply[3] = 0;
722 aReply[4] = 0;
723 aReply[5] = 0;
724 aReply[6] = 0;
725 aReply[7] = 0;
726
727 aReply[8] = 0x01;
728 aReply[9] = 0x06;
729 aReply[10] = 0x00;
730 aReply[11] = 0x05;
731 aReply[12] = 0x00;
732 aReply[13] = 0x00;
733 aReply[14] = 0x00;
734 aReply[15] = 0x00;
735 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, RT_MIN(cbMaxTransfer, sizeof(aReply)));
736 rcReq = vscsiLunReqSenseOkSet(&pVScsiLunMmc->Core, pVScsiReq);
737 break;
738 }
739 case SCSI_MODEPAGE_CD_STATUS:
740 {
741 uint8_t aReply[40];
742
743 scsiH2BE_U16(&aReply[0], 38);
744 aReply[2] = (uint8_t)pVScsiLunMmc->u32MediaTrackType;
745 aReply[3] = 0;
746 aReply[4] = 0;
747 aReply[5] = 0;
748 aReply[6] = 0;
749 aReply[7] = 0;
750
751 aReply[8] = 0x2a;
752 aReply[9] = 30; /* page length */
753 aReply[10] = 0x08; /* DVD-ROM read support */
754 aReply[11] = 0x00; /* no write support */
755 /* The following claims we support audio play. This is obviously false,
756 * but the Linux generic CDROM support makes many features depend on this
757 * capability. If it's not set, this causes many things to be disabled. */
758 aReply[12] = 0x71; /* multisession support, mode 2 form 1/2 support, audio play */
759 aReply[13] = 0x00; /* no subchannel reads supported */
760 aReply[14] = (1 << 0) | (1 << 3) | (1 << 5); /* lock supported, eject supported, tray type loading mechanism */
761 if (pVScsiLunMmc->fLocked)
762 aReply[14] |= 1 << 1; /* report lock state */
763 aReply[15] = 0; /* no subchannel reads supported, no separate audio volume control, no changer etc. */
764 scsiH2BE_U16(&aReply[16], 5632); /* (obsolete) claim 32x speed support */
765 scsiH2BE_U16(&aReply[18], 2); /* number of audio volume levels */
766 scsiH2BE_U16(&aReply[20], 128); /* buffer size supported in Kbyte - We don't have a buffer because we write directly into guest memory.
767 Just write some dummy value. */
768 scsiH2BE_U16(&aReply[22], 5632); /* (obsolete) current read speed 32x */
769 aReply[24] = 0; /* reserved */
770 aReply[25] = 0; /* reserved for digital audio (see idx 15) */
771 scsiH2BE_U16(&aReply[26], 0); /* (obsolete) maximum write speed */
772 scsiH2BE_U16(&aReply[28], 0); /* (obsolete) current write speed */
773 scsiH2BE_U16(&aReply[30], 0); /* copy management revision supported 0=no CSS */
774 aReply[32] = 0; /* reserved */
775 aReply[33] = 0; /* reserved */
776 aReply[34] = 0; /* reserved */
777 aReply[35] = 1; /* rotation control CAV */
778 scsiH2BE_U16(&aReply[36], 0); /* current write speed */
779 scsiH2BE_U16(&aReply[38], 0); /* number of write speed performance descriptors */
780 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, RT_MIN(cbMaxTransfer, sizeof(aReply)));
781 rcReq = vscsiLunReqSenseOkSet(&pVScsiLunMmc->Core, pVScsiReq);
782 break;
783 }
784 default:
785 rcReq = vscsiLunReqSenseErrorSet(&pVScsiLunMmc->Core, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
786 SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
787 break;
788 }
789 break;
790 case SCSI_PAGECONTROL_CHANGEABLE:
791 case SCSI_PAGECONTROL_DEFAULT:
792 rcReq = vscsiLunReqSenseErrorSet(&pVScsiLunMmc->Core, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
793 SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
794 break;
795 default:
796 case SCSI_PAGECONTROL_SAVED:
797 rcReq = vscsiLunReqSenseErrorSet(&pVScsiLunMmc->Core, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
798 SCSI_ASC_SAVING_PARAMETERS_NOT_SUPPORTED, 0x00);
799 break;
800 }
801
802 return rcReq;
803}
804
805/**
806 * Processes the GET EVENT STATUS NOTIFICATION SCSI request.
807 *
808 * @returns SCSI status code.
809 * @param pVScsiLunMmc The MMC LUN instance.
810 * @param pVScsiReq The VSCSI request.
811 * @param cbMaxTransfer The maximum transfer size.
812 */
813static int vscsiLunMmcGetEventStatusNotification(PVSCSILUNMMC pVScsiLunMmc, PVSCSIREQINT pVScsiReq,
814 size_t cbMaxTransfer)
815{
816 uint32_t OldStatus;
817 uint32_t NewStatus;
818 uint8_t aReply[8];
819 RT_ZERO(aReply);
820
821 LogFlowFunc(("pVScsiLunMmc=%#p pVScsiReq=%#p cbMaxTransfer=%zu\n",
822 pVScsiLunMmc, pVScsiReq, cbMaxTransfer));
823
824 do
825 {
826 OldStatus = ASMAtomicReadU32((volatile uint32_t *)&pVScsiLunMmc->MediaEventStatus);
827 NewStatus = MMCEVENTSTATUSTYPE_UNCHANGED;
828
829 switch (OldStatus)
830 {
831 case MMCEVENTSTATUSTYPE_MEDIA_NEW:
832 /* mount */
833 scsiH2BE_U16(&aReply[0], 6);
834 aReply[2] = 0x04; /* media */
835 aReply[3] = 0x5e; /* supported = busy|media|external|power|operational */
836 aReply[4] = 0x02; /* new medium */
837 aReply[5] = 0x02; /* medium present / door closed */
838 aReply[6] = 0x00;
839 aReply[7] = 0x00;
840 pVScsiLunMmc->Core.fReady = true;
841 break;
842
843 case MMCEVENTSTATUSTYPE_MEDIA_CHANGED:
844 case MMCEVENTSTATUSTYPE_MEDIA_REMOVED:
845 /* umount */
846 scsiH2BE_U16(&aReply[0], 6);
847 aReply[2] = 0x04; /* media */
848 aReply[3] = 0x5e; /* supported = busy|media|external|power|operational */
849 aReply[4] = 0x03; /* media removal */
850 aReply[5] = 0x00; /* medium absent / door closed */
851 aReply[6] = 0x00;
852 aReply[7] = 0x00;
853 if (OldStatus == MMCEVENTSTATUSTYPE_MEDIA_CHANGED)
854 NewStatus = MMCEVENTSTATUSTYPE_MEDIA_NEW;
855 break;
856
857 case MMCEVENTSTATUSTYPE_MEDIA_EJECT_REQUESTED: /* currently unused */
858 scsiH2BE_U16(&aReply[0], 6);
859 aReply[2] = 0x04; /* media */
860 aReply[3] = 0x5e; /* supported = busy|media|external|power|operational */
861 aReply[4] = 0x01; /* eject requested (eject button pressed) */
862 aReply[5] = 0x02; /* medium present / door closed */
863 aReply[6] = 0x00;
864 aReply[7] = 0x00;
865 break;
866
867 case MMCEVENTSTATUSTYPE_UNCHANGED:
868 default:
869 scsiH2BE_U16(&aReply[0], 6);
870 aReply[2] = 0x01; /* operational change request / notification */
871 aReply[3] = 0x5e; /* supported = busy|media|external|power|operational */
872 aReply[4] = 0x00;
873 aReply[5] = 0x00;
874 aReply[6] = 0x00;
875 aReply[7] = 0x00;
876 break;
877 }
878
879 LogFlowFunc(("OldStatus=%u NewStatus=%u\n", OldStatus, NewStatus));
880
881 } while (!ASMAtomicCmpXchgU32((volatile uint32_t *)&pVScsiLunMmc->MediaEventStatus, NewStatus, OldStatus));
882
883 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, RT_MIN(cbMaxTransfer, sizeof(aReply)));
884 return vscsiLunReqSenseOkSet(&pVScsiLunMmc->Core, pVScsiReq);
885}
886
887/**
888 * Processes a READ TRACK INFORMATION SCSI request.
889 *
890 * @returns SCSI status code.
891 * @param pVScsiLunMmc The MMC LUN instance.
892 * @param pVScsiReq The VSCSI request.
893 * @param cbMaxTransfer The maximum transfer size.
894 */
895static int vscsiLunMmcReadTrackInformation(PVSCSILUNMMC pVScsiLunMmc, PVSCSIREQINT pVScsiReq,
896 size_t cbMaxTransfer)
897{
898 int rcReq;
899 uint32_t u32LogAddr = scsiBE2H_U32(&pVScsiReq->pbCDB[2]);
900 uint8_t u8LogAddrType = pVScsiReq->pbCDB[1] & 0x03;
901
902 int rc = VINF_SUCCESS;
903 uint64_t u64LbaStart = 0;
904 uint32_t uRegion = 0;
905 uint64_t cBlocks = 0;
906 uint64_t cbBlock = 0;
907 VDREGIONDATAFORM enmDataForm = VDREGIONDATAFORM_INVALID;
908
909 switch (u8LogAddrType)
910 {
911 case 0x00:
912 rc = vscsiLunMediumQueryRegionPropertiesForLba(&pVScsiLunMmc->Core, u32LogAddr, &uRegion,
913 NULL, NULL, NULL);
914 if (RT_SUCCESS(rc))
915 rc = vscsiLunMediumQueryRegionProperties(&pVScsiLunMmc->Core, uRegion, &u64LbaStart,
916 &cBlocks, &cbBlock, &enmDataForm);
917 break;
918 case 0x01:
919 {
920 if (u32LogAddr >= 1)
921 {
922 uRegion = u32LogAddr - 1;
923 rc = vscsiLunMediumQueryRegionProperties(&pVScsiLunMmc->Core, uRegion, &u64LbaStart,
924 &cBlocks, &cbBlock, &enmDataForm);
925 }
926 else
927 rc = VERR_NOT_FOUND; /** @todo: Return lead-in information. */
928 break;
929 }
930 case 0x02:
931 default:
932 rc = VERR_INVALID_PARAMETER;
933 }
934
935 if (RT_SUCCESS(rc))
936 {
937 uint8_t u8DataMode = 0xf; /* Unknown data mode. */
938 uint8_t u8TrackMode = 0;
939 uint8_t aReply[36];
940 RT_ZERO(aReply);
941
942 switch (enmDataForm)
943 {
944 case VDREGIONDATAFORM_MODE1_2048:
945 case VDREGIONDATAFORM_MODE1_2352:
946 case VDREGIONDATAFORM_MODE1_0:
947 u8DataMode = 1;
948 break;
949 case VDREGIONDATAFORM_XA_2336:
950 case VDREGIONDATAFORM_XA_2352:
951 case VDREGIONDATAFORM_XA_0:
952 case VDREGIONDATAFORM_MODE2_2336:
953 case VDREGIONDATAFORM_MODE2_2352:
954 case VDREGIONDATAFORM_MODE2_0:
955 u8DataMode = 2;
956 break;
957 default:
958 u8DataMode = 0xf;
959 }
960
961 if (enmDataForm == VDREGIONDATAFORM_CDDA)
962 u8TrackMode = 0x0;
963 else
964 u8TrackMode = 0x4;
965
966 scsiH2BE_U16(&aReply[0], 34);
967 aReply[2] = uRegion + 1; /* track number (LSB) */
968 aReply[3] = 1; /* session number (LSB) */
969 aReply[5] = (0 << 5) | (0 << 4) | u8TrackMode; /* not damaged, primary copy, data track */
970 aReply[6] = (0 << 7) | (0 << 6) | (0 << 5) | (0 << 6) | u8DataMode; /* not reserved track, not blank, not packet writing, not fixed packet, data mode 1 */
971 aReply[7] = (0 << 1) | (0 << 0); /* last recorded address not valid, next recordable address not valid */
972 scsiH2BE_U32(&aReply[8], (uint32_t)u64LbaStart); /* track start address is 0 */
973 scsiH2BE_U32(&aReply[24], (uint32_t)cBlocks); /* track size */
974 aReply[32] = 0; /* track number (MSB) */
975 aReply[33] = 0; /* session number (MSB) */
976
977 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, RT_MIN(sizeof(aReply), cbMaxTransfer));
978 rcReq = vscsiLunReqSenseOkSet(&pVScsiLunMmc->Core, pVScsiReq);
979 }
980 else
981 rcReq = vscsiLunReqSenseErrorSet(&pVScsiLunMmc->Core, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
982 SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
983
984 return rcReq;
985}
986
987static DECLCALLBACK(int) vscsiLunMmcInit(PVSCSILUNINT pVScsiLun)
988{
989 PVSCSILUNMMC pVScsiLunMmc = (PVSCSILUNMMC)pVScsiLun;
990 int rc = VINF_SUCCESS;
991
992 ASMAtomicWriteU32((volatile uint32_t *)&pVScsiLunMmc->MediaEventStatus, MMCEVENTSTATUSTYPE_UNCHANGED);
993 pVScsiLunMmc->u32MediaTrackType = MMC_MEDIA_TYPE_UNKNOWN;
994 pVScsiLunMmc->cSectors = 0;
995
996 uint32_t cTracks = vscsiLunMediumGetRegionCount(pVScsiLun);
997 for (uint32_t i = 0; i < cTracks; i++)
998 {
999 uint64_t cBlocks = 0;
1000 rc = vscsiLunMediumQueryRegionProperties(pVScsiLun, i, NULL, &cBlocks,
1001 NULL, NULL);
1002 AssertRC(rc);
1003
1004 pVScsiLunMmc->cSectors += cBlocks;
1005 }
1006
1007 return rc;
1008}
1009
1010static DECLCALLBACK(int) vscsiLunMmcDestroy(PVSCSILUNINT pVScsiLun)
1011{
1012 RT_NOREF1(pVScsiLun);
1013 return VINF_SUCCESS;
1014}
1015
1016static DECLCALLBACK(int) vscsiLunMmcReqProcess(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq)
1017{
1018 PVSCSILUNMMC pVScsiLunMmc = (PVSCSILUNMMC)pVScsiLun;
1019 VSCSIIOREQTXDIR enmTxDir = VSCSIIOREQTXDIR_INVALID;
1020 uint64_t uLbaStart = 0;
1021 uint32_t cSectorTransfer = 0;
1022 size_t cbSector = 0;
1023 int rc = VINF_SUCCESS;
1024 int rcReq = SCSI_STATUS_OK;
1025 unsigned uCmd = pVScsiReq->pbCDB[0];
1026 PCRTSGSEG paSegs = pVScsiReq->SgBuf.paSegs;
1027 unsigned cSegs = pVScsiReq->SgBuf.cSegs;
1028
1029 LogFlowFunc(("pVScsiLun=%#p{.fReady=%RTbool, .fMediaPresent=%RTbool} pVScsiReq=%#p{.pbCdb[0]=%#x}\n",
1030 pVScsiLun, pVScsiLun->fReady, pVScsiLun->fMediaPresent, pVScsiReq, uCmd));
1031
1032 /*
1033 * GET CONFIGURATION, GET EVENT/STATUS NOTIFICATION, INQUIRY, and REQUEST SENSE commands
1034 * operate even when a unit attention condition exists for initiator; every other command
1035 * needs to report CHECK CONDITION in that case.
1036 */
1037 if ( !pVScsiLunMmc->Core.fReady
1038 && uCmd != SCSI_INQUIRY
1039 && uCmd != SCSI_GET_CONFIGURATION
1040 && uCmd != SCSI_GET_EVENT_STATUS_NOTIFICATION)
1041 {
1042 /*
1043 * A note on media changes: As long as a medium is not present, the unit remains in
1044 * the 'not ready' state. Technically the unit becomes 'ready' soon after a medium
1045 * is inserted; however, we internally keep the 'not ready' state until we've had
1046 * a chance to report the UNIT ATTENTION status indicating a media change.
1047 */
1048 if (pVScsiLunMmc->Core.fMediaPresent)
1049 {
1050 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_UNIT_ATTENTION,
1051 SCSI_ASC_MEDIUM_MAY_HAVE_CHANGED, 0x00);
1052 pVScsiLunMmc->Core.fReady = true;
1053 }
1054 else
1055 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_NOT_READY,
1056 SCSI_ASC_MEDIUM_NOT_PRESENT, 0x00);
1057 }
1058 else
1059 {
1060 switch (uCmd)
1061 {
1062 case SCSI_TEST_UNIT_READY:
1063 Assert(!pVScsiLunMmc->Core.fReady); /* Only should get here if LUN isn't ready. */
1064 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_NOT_READY, SCSI_ASC_MEDIUM_NOT_PRESENT, 0x00);
1065 break;
1066
1067 case SCSI_INQUIRY:
1068 {
1069 SCSIINQUIRYDATA ScsiInquiryReply;
1070
1071 vscsiReqSetXferSize(pVScsiReq, scsiBE2H_U16(&pVScsiReq->pbCDB[3]));
1072 memset(&ScsiInquiryReply, 0, sizeof(ScsiInquiryReply));
1073
1074 ScsiInquiryReply.cbAdditional = 31;
1075 ScsiInquiryReply.fRMB = 1; /* Removable. */
1076 ScsiInquiryReply.u5PeripheralDeviceType = SCSI_INQUIRY_DATA_PERIPHERAL_DEVICE_TYPE_CD_DVD;
1077 ScsiInquiryReply.u3PeripheralQualifier = SCSI_INQUIRY_DATA_PERIPHERAL_QUALIFIER_CONNECTED;
1078 ScsiInquiryReply.u3AnsiVersion = 0x05; /* MMC-?? compliant */
1079 ScsiInquiryReply.fCmdQue = 1; /* Command queuing supported. */
1080 ScsiInquiryReply.fWBus16 = 1;
1081 scsiPadStrS(ScsiInquiryReply.achVendorId, "VBOX", 8);
1082 scsiPadStrS(ScsiInquiryReply.achProductId, "CD-ROM", 16);
1083 scsiPadStrS(ScsiInquiryReply.achProductLevel, "1.0", 4);
1084
1085 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, (uint8_t *)&ScsiInquiryReply, sizeof(SCSIINQUIRYDATA));
1086 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1087 break;
1088 }
1089 case SCSI_READ_CAPACITY:
1090 {
1091 uint8_t aReply[8];
1092 memset(aReply, 0, sizeof(aReply));
1093 vscsiReqSetXferSize(pVScsiReq, sizeof(aReply));
1094
1095 /*
1096 * If sector size exceeds the maximum value that is
1097 * able to be stored in 4 bytes return 0xffffffff in this field
1098 */
1099 if (pVScsiLunMmc->cSectors > UINT32_C(0xffffffff))
1100 scsiH2BE_U32(aReply, UINT32_C(0xffffffff));
1101 else
1102 scsiH2BE_U32(aReply, pVScsiLunMmc->cSectors - 1);
1103 scsiH2BE_U32(&aReply[4], _2K);
1104 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
1105 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1106 break;
1107 }
1108 case SCSI_MODE_SENSE_6:
1109 {
1110 uint8_t uModePage = pVScsiReq->pbCDB[2] & 0x3f;
1111 uint8_t aReply[24];
1112 uint8_t *pu8ReplyPos;
1113 bool fValid = false;
1114
1115 vscsiReqSetXferSize(pVScsiReq, pVScsiReq->pbCDB[4]);
1116 memset(aReply, 0, sizeof(aReply));
1117 aReply[0] = 4; /* Reply length 4. */
1118 aReply[1] = 0; /* Default media type. */
1119 aReply[2] = RT_BIT(4); /* Caching supported. */
1120 aReply[3] = 0; /* Block descriptor length. */
1121
1122 pu8ReplyPos = aReply + 4;
1123
1124 if ((uModePage == 0x08) || (uModePage == 0x3f))
1125 {
1126 memset(pu8ReplyPos, 0, 20);
1127 *pu8ReplyPos++ = 0x08; /* Page code. */
1128 *pu8ReplyPos++ = 0x12; /* Size of the page. */
1129 *pu8ReplyPos++ = 0x4; /* Write cache enabled. */
1130 fValid = true;
1131 } else if (uModePage == 0) {
1132 fValid = true;
1133 }
1134
1135 /* Querying unknown pages must fail. */
1136 if (fValid) {
1137 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
1138 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1139 } else {
1140 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
1141 }
1142 break;
1143 }
1144 case SCSI_MODE_SENSE_10:
1145 {
1146 size_t cbMax = scsiBE2H_U16(&pVScsiReq->pbCDB[7]);
1147 vscsiReqSetXferSize(pVScsiReq, cbMax);
1148 rcReq = vscsiLunMmcModeSense10(pVScsiLunMmc, pVScsiReq, cbMax);
1149 break;
1150 }
1151 case SCSI_SEEK_10:
1152 {
1153 uint32_t uLba = scsiBE2H_U32(&pVScsiReq->pbCDB[2]);
1154 if (uLba > pVScsiLunMmc->cSectors)
1155 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
1156 SCSI_ASC_LOGICAL_BLOCK_OOR, 0x00);
1157 else
1158 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1159 break;
1160 }
1161 case SCSI_MODE_SELECT_6:
1162 {
1163 /** @todo implement!! */
1164 vscsiReqSetXferSize(pVScsiReq, pVScsiReq->pbCDB[4]);
1165 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1166 break;
1167 }
1168 case SCSI_READ_6:
1169 {
1170 enmTxDir = VSCSIIOREQTXDIR_READ;
1171 uLbaStart = ((uint64_t) pVScsiReq->pbCDB[3]
1172 | (pVScsiReq->pbCDB[2] << 8)
1173 | ((pVScsiReq->pbCDB[1] & 0x1f) << 16));
1174 cSectorTransfer = pVScsiReq->pbCDB[4];
1175 cbSector = _2K;
1176 break;
1177 }
1178 case SCSI_READ_10:
1179 {
1180 enmTxDir = VSCSIIOREQTXDIR_READ;
1181 uLbaStart = scsiBE2H_U32(&pVScsiReq->pbCDB[2]);
1182 cSectorTransfer = scsiBE2H_U16(&pVScsiReq->pbCDB[7]);
1183 cbSector = _2K;
1184 break;
1185 }
1186 case SCSI_READ_12:
1187 {
1188 enmTxDir = VSCSIIOREQTXDIR_READ;
1189 uLbaStart = scsiBE2H_U32(&pVScsiReq->pbCDB[2]);
1190 cSectorTransfer = scsiBE2H_U32(&pVScsiReq->pbCDB[6]);
1191 cbSector = _2K;
1192 break;
1193 }
1194 case SCSI_READ_16:
1195 {
1196 enmTxDir = VSCSIIOREQTXDIR_READ;
1197 uLbaStart = scsiBE2H_U64(&pVScsiReq->pbCDB[2]);
1198 cSectorTransfer = scsiBE2H_U32(&pVScsiReq->pbCDB[10]);
1199 cbSector = _2K;
1200 break;
1201 }
1202 case SCSI_READ_CD:
1203 {
1204 uLbaStart = scsiBE2H_U32(&pVScsiReq->pbCDB[2]);
1205 cSectorTransfer = (pVScsiReq->pbCDB[6] << 16) | (pVScsiReq->pbCDB[7] << 8) | pVScsiReq->pbCDB[8];
1206
1207 /*
1208 * If the LBA is in an audio track we are required to ignore pretty much all
1209 * of the channel selection values (except 0x00) and map everything to 0x10
1210 * which means read user data with a sector size of 2352 bytes.
1211 *
1212 * (MMC-6 chapter 6.19.2.6)
1213 */
1214 uint8_t uChnSel = pVScsiReq->pbCDB[9] & 0xf8;
1215 VDREGIONDATAFORM enmDataForm;
1216 uint64_t cbSectorRegion = 0;
1217 rc = vscsiLunMediumQueryRegionPropertiesForLba(pVScsiLun, uLbaStart,
1218 NULL, NULL, &cbSectorRegion,
1219 &enmDataForm);
1220 AssertRC(rc);
1221
1222 if (enmDataForm == VDREGIONDATAFORM_CDDA)
1223 {
1224 if (uChnSel == 0)
1225 {
1226 /* nothing */
1227 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1228 }
1229 else
1230 {
1231 enmTxDir = VSCSIIOREQTXDIR_READ;
1232 cbSector = 2352;
1233 }
1234 }
1235 else
1236 {
1237 switch (uChnSel)
1238 {
1239 case 0x00:
1240 /* nothing */
1241 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1242 break;
1243 case 0x10:
1244 /* normal read */
1245 enmTxDir = VSCSIIOREQTXDIR_READ;
1246 cbSector = _2K;
1247 break;
1248 case 0xf8:
1249 {
1250 if (cbSectorRegion == 2048)
1251 {
1252 /*
1253 * Read all data, sector size is 2352.
1254 * Rearrange the buffer and fill the gaps with the sync bytes.
1255 */
1256 /* Count the number of segments for the buffer we require. */
1257 RTSGBUF SgBuf;
1258 bool fBufTooSmall = false;
1259 uint32_t cSegsNew = 0;
1260 RTSgBufClone(&SgBuf, &pVScsiReq->SgBuf);
1261 for (uint32_t uLba = (uint32_t)uLbaStart; uLba < uLbaStart + cSectorTransfer; uLba++)
1262 {
1263 size_t cbTmp = RTSgBufAdvance(&SgBuf, 16);
1264 if (cbTmp < 16)
1265 {
1266 fBufTooSmall = true;
1267 break;
1268 }
1269
1270 cbTmp = 2048;
1271 while (cbTmp)
1272 {
1273 size_t cbBuf = cbTmp;
1274 RTSgBufGetNextSegment(&SgBuf, &cbBuf);
1275 if (!cbBuf)
1276 {
1277 fBufTooSmall = true;
1278 break;
1279 }
1280
1281 cbTmp -= cbBuf;
1282 cSegsNew++;
1283 }
1284
1285 cbTmp = RTSgBufAdvance(&SgBuf, 280);
1286 if (cbTmp < 280)
1287 {
1288 fBufTooSmall = true;
1289 break;
1290 }
1291 }
1292
1293 if (!fBufTooSmall)
1294 {
1295 PRTSGSEG paSegsNew = (PRTSGSEG)RTMemAllocZ(cSegsNew * sizeof(RTSGSEG));
1296 if (paSegsNew)
1297 {
1298 enmTxDir = VSCSIIOREQTXDIR_READ;
1299
1300 uint32_t idxSeg = 0;
1301 for (uint32_t uLba = (uint32_t)uLbaStart; uLba < uLbaStart + cSectorTransfer; uLba++)
1302 {
1303 /* Sync bytes, see 4.2.3.8 CD Main Channel Block Formats */
1304 uint8_t abBuf[16];
1305 abBuf[0] = 0x00;
1306 memset(&abBuf[1], 0xff, 10);
1307 abBuf[11] = 0x00;
1308 /* MSF */
1309 scsiLBA2MSF(&abBuf[12], uLba);
1310 abBuf[15] = 0x01; /* mode 1 data */
1311 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, &abBuf[0], sizeof(abBuf));
1312
1313 size_t cbTmp = 2048;
1314 while (cbTmp)
1315 {
1316 size_t cbBuf = cbTmp;
1317 paSegsNew[idxSeg].pvSeg = RTSgBufGetNextSegment(&pVScsiReq->SgBuf, &cbBuf);
1318 paSegsNew[idxSeg].cbSeg = cbBuf;
1319 idxSeg++;
1320
1321 cbTmp -= cbBuf;
1322 }
1323
1324 /**
1325 * @todo: maybe compute ECC and parity, layout is:
1326 * 2072 4 EDC
1327 * 2076 172 P parity symbols
1328 * 2248 104 Q parity symbols
1329 */
1330 RTSgBufSet(&pVScsiReq->SgBuf, 0, 280);
1331 }
1332
1333 paSegs = paSegsNew;
1334 cSegs = cSegsNew;
1335 pVScsiReq->pvLun = paSegsNew;
1336 }
1337 else
1338 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
1339 SCSI_ASC_LOGICAL_BLOCK_OOR, 0x00);
1340 }
1341 else
1342 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
1343 SCSI_ASC_LOGICAL_BLOCK_OOR, 0x00);
1344 }
1345 else if (cbSectorRegion == 2352)
1346 {
1347 /* Sector size matches what is read. */
1348 cbSector = cbSectorRegion;
1349 enmTxDir = VSCSIIOREQTXDIR_READ;
1350 }
1351 break;
1352 }
1353 default:
1354 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
1355 SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
1356 break;
1357 }
1358 }
1359 break;
1360 }
1361 case SCSI_READ_BUFFER:
1362 {
1363 uint8_t uDataMode = pVScsiReq->pbCDB[1] & 0x1f;
1364
1365 vscsiReqSetXferSize(pVScsiReq, scsiBE2H_U16(&pVScsiReq->pbCDB[6]));
1366
1367 switch (uDataMode)
1368 {
1369 case 0x00:
1370 case 0x01:
1371 case 0x02:
1372 case 0x03:
1373 case 0x0a:
1374 break;
1375 case 0x0b:
1376 {
1377 uint8_t aReply[4];
1378 RT_ZERO(aReply);
1379
1380 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
1381 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1382 break;
1383 }
1384 case 0x1a:
1385 case 0x1c:
1386 break;
1387 default:
1388 AssertMsgFailed(("Invalid data mode\n"));
1389 }
1390 break;
1391 }
1392 case SCSI_VERIFY_10:
1393 case SCSI_START_STOP_UNIT:
1394 {
1395 int rc2 = VINF_SUCCESS;
1396 switch (pVScsiReq->pbCDB[4] & 3)
1397 {
1398 case 0: /* 00 - Stop motor */
1399 case 1: /* 01 - Start motor */
1400 break;
1401 case 2: /* 10 - Eject media */
1402 rc2 = vscsiLunMediumEject(pVScsiLun);
1403 break;
1404 case 3: /* 11 - Load media */
1405 /** @todo */
1406 break;
1407 }
1408 if (RT_SUCCESS(rc2))
1409 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1410 else
1411 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_MEDIA_LOAD_OR_EJECT_FAILED, 0x02);
1412 break;
1413 }
1414 case SCSI_LOG_SENSE:
1415 {
1416 uint8_t uPageCode = pVScsiReq->pbCDB[2] & 0x3f;
1417 uint8_t uSubPageCode = pVScsiReq->pbCDB[3];
1418
1419 vscsiReqSetXferSize(pVScsiReq, scsiBE2H_U16(&pVScsiReq->pbCDB[7]));
1420
1421 switch (uPageCode)
1422 {
1423 case 0x00:
1424 {
1425 if (uSubPageCode == 0)
1426 {
1427 uint8_t aReply[4];
1428
1429 aReply[0] = 0;
1430 aReply[1] = 0;
1431 aReply[2] = 0;
1432 aReply[3] = 0;
1433
1434 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
1435 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1436 break;
1437 }
1438 }
1439 /* fall thru */
1440 default:
1441 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
1442 }
1443 break;
1444 }
1445 case SCSI_SERVICE_ACTION_IN_16:
1446 {
1447 switch (pVScsiReq->pbCDB[1] & 0x1f)
1448 {
1449 case SCSI_SVC_ACTION_IN_READ_CAPACITY_16:
1450 {
1451 uint8_t aReply[32];
1452
1453 memset(aReply, 0, sizeof(aReply));
1454 scsiH2BE_U64(aReply, pVScsiLunMmc->cSectors - 1);
1455 scsiH2BE_U32(&aReply[8], _2K);
1456 /* Leave the rest 0 */
1457
1458 vscsiReqSetXferSize(pVScsiReq, sizeof(aReply));
1459 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
1460 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1461 break;
1462 }
1463 default:
1464 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST,
1465 SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00); /* Don't know if this is correct */
1466 }
1467 break;
1468 }
1469 case SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL:
1470 {
1471 pVScsiLunMmc->fLocked = RT_BOOL(pVScsiReq->pbCDB[4] & 0x01);
1472 vscsiLunMediumSetLock(pVScsiLun, pVScsiLunMmc->fLocked);
1473 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1474 break;
1475 }
1476 case SCSI_READ_TOC_PMA_ATIP:
1477 {
1478 uint8_t format;
1479 uint16_t cbMax;
1480 bool fMSF;
1481
1482 format = pVScsiReq->pbCDB[2] & 0x0f;
1483 cbMax = scsiBE2H_U16(&pVScsiReq->pbCDB[7]);
1484 fMSF = (pVScsiReq->pbCDB[1] >> 1) & 1;
1485
1486 vscsiReqSetXferSize(pVScsiReq, cbMax);
1487 switch (format)
1488 {
1489 case 0x00:
1490 rcReq = mmcReadTOCNormal(pVScsiLun, pVScsiReq, cbMax, fMSF);
1491 break;
1492 case 0x01:
1493 rcReq = mmcReadTOCMulti(pVScsiLun, pVScsiReq, cbMax, fMSF);
1494 break;
1495 case 0x02:
1496 rcReq = mmcReadTOCRaw(pVScsiLun, pVScsiReq, cbMax, fMSF);
1497 break;
1498 default:
1499 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
1500 }
1501 break;
1502 }
1503 case SCSI_GET_EVENT_STATUS_NOTIFICATION:
1504 {
1505 /* Only supporting polled mode at the moment. */
1506 size_t cbMax = scsiBE2H_U16(&pVScsiReq->pbCDB[7]);
1507
1508 vscsiReqSetXferSize(pVScsiReq, cbMax);
1509 if (pVScsiReq->pbCDB[1] & 0x1)
1510 rcReq = vscsiLunMmcGetEventStatusNotification(pVScsiLunMmc, pVScsiReq, cbMax);
1511 else
1512 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
1513 break;
1514 }
1515 case SCSI_MECHANISM_STATUS:
1516 {
1517 size_t cbMax = scsiBE2H_U16(&pVScsiReq->pbCDB[8]);
1518 uint8_t aReply[8];
1519
1520 vscsiReqSetXferSize(pVScsiReq, cbMax);
1521 scsiH2BE_U16(&aReply[0], 0);
1522 /* no current LBA */
1523 aReply[2] = 0;
1524 aReply[3] = 0;
1525 aReply[4] = 0;
1526 aReply[5] = 1;
1527 scsiH2BE_U16(&aReply[6], 0);
1528 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, RT_MIN(sizeof(aReply), cbMax));
1529 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1530 break;
1531 }
1532 case SCSI_READ_DISC_INFORMATION:
1533 {
1534 uint8_t aReply[34];
1535 size_t cbMax = scsiBE2H_U16(&pVScsiReq->pbCDB[7]);
1536
1537 vscsiReqSetXferSize(pVScsiReq, cbMax);
1538 memset(aReply, '\0', sizeof(aReply));
1539 scsiH2BE_U16(&aReply[0], 32);
1540 aReply[2] = (0 << 4) | (3 << 2) | (2 << 0); /* not erasable, complete session, complete disc */
1541 aReply[3] = 1; /* number of first track */
1542 aReply[4] = 1; /* number of sessions (LSB) */
1543 aReply[5] = 1; /* first track number in last session (LSB) */
1544 aReply[6] = (uint8_t)vscsiLunMediumGetRegionCount(pVScsiLun); /* last track number in last session (LSB) */
1545 aReply[7] = (0 << 7) | (0 << 6) | (1 << 5) | (0 << 2) | (0 << 0); /* disc id not valid, disc bar code not valid, unrestricted use, not dirty, not RW medium */
1546 aReply[8] = 0; /* disc type = CD-ROM */
1547 aReply[9] = 0; /* number of sessions (MSB) */
1548 aReply[10] = 0; /* number of sessions (MSB) */
1549 aReply[11] = 0; /* number of sessions (MSB) */
1550 scsiH2BE_U32(&aReply[16], 0x00ffffff); /* last session lead-in start time is not available */
1551 scsiH2BE_U32(&aReply[20], 0x00ffffff); /* last possible start time for lead-out is not available */
1552 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, RT_MIN(sizeof(aReply), cbMax));
1553 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1554 break;
1555 }
1556 case SCSI_READ_TRACK_INFORMATION:
1557 {
1558 size_t cbMax = scsiBE2H_U16(&pVScsiReq->pbCDB[7]);
1559 vscsiReqSetXferSize(pVScsiReq, cbMax);
1560 rcReq = vscsiLunMmcReadTrackInformation(pVScsiLunMmc, pVScsiReq, cbMax);
1561 break;
1562 }
1563 case SCSI_GET_CONFIGURATION:
1564 {
1565 size_t cbMax = scsiBE2H_U16(&pVScsiReq->pbCDB[7]);
1566 vscsiReqSetXferSize(pVScsiReq, cbMax);
1567 rcReq = vscsiLunMmcGetConfiguration(pVScsiLunMmc, pVScsiReq, cbMax);
1568 break;
1569 }
1570 case SCSI_READ_DVD_STRUCTURE:
1571 {
1572 size_t cbMax = scsiBE2H_U16(&pVScsiReq->pbCDB[8]);
1573 vscsiReqSetXferSize(pVScsiReq, cbMax);
1574 rcReq = vscsiLunMmcReadDvdStructure(pVScsiLunMmc, pVScsiReq, cbMax);
1575 break;
1576 }
1577 default:
1578 //AssertMsgFailed(("Command %#x [%s] not implemented\n", pVScsiReq->pbCDB[0], SCSICmdText(pVScsiReq->pbCDB[0])));
1579 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE, 0x00);
1580 }
1581 }
1582
1583 if (enmTxDir != VSCSIIOREQTXDIR_INVALID)
1584 {
1585 LogFlow(("%s: uLbaStart=%llu cSectorTransfer=%u\n",
1586 __FUNCTION__, uLbaStart, cSectorTransfer));
1587
1588 vscsiReqSetXferSize(pVScsiReq, cSectorTransfer * cbSector);
1589 if (RT_UNLIKELY(uLbaStart + cSectorTransfer > pVScsiLunMmc->cSectors))
1590 {
1591 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_LOGICAL_BLOCK_OOR, 0x00);
1592 vscsiDeviceReqComplete(pVScsiLun->pVScsiDevice, pVScsiReq, rcReq, false, VINF_SUCCESS);
1593 }
1594 else if (!cSectorTransfer)
1595 {
1596 /* A 0 transfer length is not an error. */
1597 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
1598 vscsiDeviceReqComplete(pVScsiLun->pVScsiDevice, pVScsiReq, rcReq, false, VINF_SUCCESS);
1599 }
1600 else
1601 {
1602 /* Check that the sector size is valid. */
1603 VDREGIONDATAFORM enmDataForm = VDREGIONDATAFORM_INVALID;
1604 rc = vscsiLunMediumQueryRegionPropertiesForLba(pVScsiLun, uLbaStart,
1605 NULL, NULL, NULL, &enmDataForm);
1606 AssertRC(rc);
1607 if ( enmDataForm != VDREGIONDATAFORM_MODE1_2048
1608 && enmDataForm != VDREGIONDATAFORM_MODE1_2352
1609 && enmDataForm != VDREGIONDATAFORM_MODE2_2336
1610 && enmDataForm != VDREGIONDATAFORM_MODE2_2352
1611 && enmDataForm != VDREGIONDATAFORM_RAW
1612 && cbSector == _2K)
1613 {
1614 rcReq = vscsiLunReqSenseErrorInfoSet(pVScsiLun, pVScsiReq,
1615 SCSI_SENSE_ILLEGAL_REQUEST | SCSI_SENSE_FLAG_ILI,
1616 SCSI_ASC_ILLEGAL_MODE_FOR_THIS_TRACK, 0, uLbaStart);
1617 vscsiDeviceReqComplete(pVScsiLun->pVScsiDevice, pVScsiReq, rcReq, false, VINF_SUCCESS);
1618 }
1619 else
1620 {
1621 /* Enqueue new I/O request */
1622 rc = vscsiIoReqTransferEnqueueEx(pVScsiLun, pVScsiReq, enmTxDir,
1623 uLbaStart * cbSector,
1624 paSegs, cSegs, cSectorTransfer * cbSector);
1625 }
1626 }
1627 }
1628 else /* Request completed */
1629 vscsiDeviceReqComplete(pVScsiLun->pVScsiDevice, pVScsiReq, rcReq, false, VINF_SUCCESS);
1630
1631 return rc;
1632}
1633
1634/** @interface_method_impl{VSCSILUNDESC,pfnVScsiLunReqFree} */
1635static DECLCALLBACK(void) vscsiLunMmcReqFree(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq,
1636 void *pvLun)
1637{
1638 RT_NOREF2(pVScsiLun, pVScsiReq);
1639 RTMemFree(pvLun);
1640}
1641
1642/** @interface_method_impl{VSCSILUNDESC,pfnVScsiLunMediumInserted} */
1643static DECLCALLBACK(int) vscsiLunMmcMediumInserted(PVSCSILUNINT pVScsiLun)
1644{
1645 int rc = VINF_SUCCESS;
1646 PVSCSILUNMMC pVScsiLunMmc = (PVSCSILUNMMC)pVScsiLun;
1647
1648 pVScsiLunMmc->cSectors = 0;
1649 uint32_t cTracks = vscsiLunMediumGetRegionCount(pVScsiLun);
1650 for (uint32_t i = 0; i < cTracks && RT_SUCCESS(rc); i++)
1651 {
1652 uint64_t cBlocks = 0;
1653 rc = vscsiLunMediumQueryRegionProperties(pVScsiLun, i, NULL, &cBlocks,
1654 NULL, NULL);
1655 if (RT_FAILURE(rc))
1656 break;
1657 pVScsiLunMmc->cSectors += cBlocks;
1658 }
1659
1660 if (RT_SUCCESS(rc))
1661 {
1662 uint32_t OldStatus, NewStatus;
1663 do
1664 {
1665 OldStatus = ASMAtomicReadU32((volatile uint32_t *)&pVScsiLunMmc->MediaEventStatus);
1666 switch (OldStatus)
1667 {
1668 case MMCEVENTSTATUSTYPE_MEDIA_CHANGED:
1669 case MMCEVENTSTATUSTYPE_MEDIA_REMOVED:
1670 /* no change, we will send "medium removed" + "medium inserted" */
1671 NewStatus = MMCEVENTSTATUSTYPE_MEDIA_CHANGED;
1672 break;
1673 default:
1674 NewStatus = MMCEVENTSTATUSTYPE_MEDIA_NEW;
1675 break;
1676 }
1677 } while (!ASMAtomicCmpXchgU32((volatile uint32_t *)&pVScsiLunMmc->MediaEventStatus,
1678 NewStatus, OldStatus));
1679
1680 ASMAtomicXchgU32(&pVScsiLunMmc->u32MediaTrackType, MMC_MEDIA_TYPE_UNKNOWN);
1681 }
1682
1683 return rc;
1684}
1685
1686/** @interface_method_impl{VSCSILUNDESC,pfnVScsiLunMediumRemoved} */
1687static DECLCALLBACK(int) vscsiLunMmcMediumRemoved(PVSCSILUNINT pVScsiLun)
1688{
1689 PVSCSILUNMMC pVScsiLunMmc = (PVSCSILUNMMC)pVScsiLun;
1690
1691 ASMAtomicWriteU32((volatile uint32_t *)&pVScsiLunMmc->MediaEventStatus, MMCEVENTSTATUSTYPE_MEDIA_REMOVED);
1692 ASMAtomicXchgU32(&pVScsiLunMmc->u32MediaTrackType, MMC_MEDIA_TYPE_NO_DISC);
1693 return VINF_SUCCESS;
1694}
1695
1696
1697VSCSILUNDESC g_VScsiLunTypeMmc =
1698{
1699 /** enmLunType */
1700 VSCSILUNTYPE_MMC,
1701 /** pcszDescName */
1702 "MMC",
1703 /** cbLun */
1704 sizeof(VSCSILUNMMC),
1705 /** cSupOpcInfo */
1706 0,
1707 /** paSupOpcInfo */
1708 NULL,
1709 /** pfnVScsiLunInit */
1710 vscsiLunMmcInit,
1711 /** pfnVScsiLunDestroy */
1712 vscsiLunMmcDestroy,
1713 /** pfnVScsiLunReqProcess */
1714 vscsiLunMmcReqProcess,
1715 /** pfnVScsiLunReqFree */
1716 vscsiLunMmcReqFree,
1717 /** pfnVScsiLunMediumInserted */
1718 vscsiLunMmcMediumInserted,
1719 /** pfnVScsiLunMediumRemoved */
1720 vscsiLunMmcMediumRemoved
1721};
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