VirtualBox

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

Last change on this file since 42045 was 40500, checked in by vboxsync, 13 years ago

Keep track of medium lock state.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 11.3 KB
Line 
1/* $Id: VSCSILunMmc.cpp 40500 2012-03-16 14:40:07Z vboxsync $ */
2/** @file
3 * Virtual SCSI driver: MMC LUN implementation (CD/DVD-ROM)
4 */
5
6/*
7 * Copyright (C) 2006-2011 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/*******************************************************************************
19* Header Files *
20*******************************************************************************/
21#define LOG_GROUP LOG_GROUP_VSCSI
22#include <VBox/log.h>
23#include <VBox/err.h>
24#include <VBox/types.h>
25#include <VBox/vscsi.h>
26#include <iprt/assert.h>
27#include <iprt/mem.h>
28#include <iprt/string.h>
29
30#include "VSCSIInternal.h"
31
32/**
33 * MMC LUN instance
34 */
35typedef struct VSCSILUNMMC
36{
37 /** Core LUN structure */
38 VSCSILUNINT Core;
39 /** Size of the virtual disk. */
40 uint64_t cSectors;
41 /** Sector size. */
42 uint32_t cbSector;
43 /** Medium locked indicator. */
44 bool fLocked;
45} VSCSILUNMMC, *PVSCSILUNMMC;
46
47static int vscsiLunMmcInit(PVSCSILUNINT pVScsiLun)
48{
49 PVSCSILUNMMC pVScsiLunMmc = (PVSCSILUNMMC)pVScsiLun;
50 uint64_t cbDisk = 0;
51 int rc = VINF_SUCCESS;
52
53 pVScsiLunMmc->cbSector = 2048; /* Default to 2K sectors. */
54 rc = vscsiLunMediumGetSize(pVScsiLun, &cbDisk);
55 if (RT_SUCCESS(rc))
56 pVScsiLunMmc->cSectors = cbDisk / pVScsiLunMmc->cbSector;
57
58 return rc;
59}
60
61static int vscsiLunMmcDestroy(PVSCSILUNINT pVScsiLun)
62{
63 PVSCSILUNMMC pVScsiLunMmc = (PVSCSILUNMMC)pVScsiLun;
64
65 return VINF_SUCCESS;
66}
67
68static int vscsiLunMmcReqProcess(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq)
69{
70 PVSCSILUNMMC pVScsiLunMmc = (PVSCSILUNMMC)pVScsiLun;
71 VSCSIIOREQTXDIR enmTxDir = VSCSIIOREQTXDIR_INVALID;
72 uint64_t uLbaStart = 0;
73 uint32_t cSectorTransfer = 0;
74 int rc = VINF_SUCCESS;
75 int rcReq = SCSI_STATUS_OK;
76
77 switch(pVScsiReq->pbCDB[0])
78 {
79 case SCSI_INQUIRY:
80 {
81 SCSIINQUIRYDATA ScsiInquiryReply;
82
83 memset(&ScsiInquiryReply, 0, sizeof(ScsiInquiryReply));
84
85 ScsiInquiryReply.cbAdditional = 31;
86 ScsiInquiryReply.fRMB = 1; /* Removable. */
87 ScsiInquiryReply.u5PeripheralDeviceType = SCSI_INQUIRY_DATA_PERIPHERAL_DEVICE_TYPE_CD_DVD;
88 ScsiInquiryReply.u3PeripheralQualifier = SCSI_INQUIRY_DATA_PERIPHERAL_QUALIFIER_CONNECTED;
89 ScsiInquiryReply.u3AnsiVersion = 0x05; /* MMC-?? compliant */
90 ScsiInquiryReply.fCmdQue = 1; /* Command queuing supported. */
91 ScsiInquiryReply.fWBus16 = 1;
92 vscsiPadStr(ScsiInquiryReply.achVendorId, "VBOX", 8);
93 vscsiPadStr(ScsiInquiryReply.achProductId, "CD-ROM", 16);
94 vscsiPadStr(ScsiInquiryReply.achProductLevel, "1.0", 4);
95
96 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, (uint8_t *)&ScsiInquiryReply, sizeof(SCSIINQUIRYDATA));
97 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
98 break;
99 }
100 case SCSI_READ_CAPACITY:
101 {
102 uint8_t aReply[8];
103 memset(aReply, 0, sizeof(aReply));
104
105 /*
106 * If sector size exceeds the maximum value that is
107 * able to be stored in 4 bytes return 0xffffffff in this field
108 */
109 if (pVScsiLunMmc->cSectors > UINT32_C(0xffffffff))
110 vscsiH2BEU32(aReply, UINT32_C(0xffffffff));
111 else
112 vscsiH2BEU32(aReply, pVScsiLunMmc->cSectors - 1);
113 vscsiH2BEU32(&aReply[4], pVScsiLunMmc->cbSector);
114 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
115 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
116 break;
117 }
118 case SCSI_MODE_SENSE_6:
119 {
120 uint8_t uModePage = pVScsiReq->pbCDB[2] & 0x3f;
121 uint8_t aReply[24];
122 uint8_t *pu8ReplyPos;
123
124 memset(aReply, 0, sizeof(aReply));
125 aReply[0] = 4; /* Reply length 4. */
126 aReply[1] = 0; /* Default media type. */
127 aReply[2] = RT_BIT(4); /* Caching supported. */
128 aReply[3] = 0; /* Block descriptor length. */
129
130 pu8ReplyPos = aReply + 4;
131
132 if ((uModePage == 0x08) || (uModePage == 0x3f))
133 {
134 memset(pu8ReplyPos, 0, 20);
135 *pu8ReplyPos++ = 0x08; /* Page code. */
136 *pu8ReplyPos++ = 0x12; /* Size of the page. */
137 *pu8ReplyPos++ = 0x4; /* Write cache enabled. */
138 }
139
140 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
141 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
142 break;
143 }
144 case SCSI_MODE_SELECT_6:
145 {
146 /* @todo: implement!! */
147 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
148 break;
149 }
150 case SCSI_READ_6:
151 {
152 enmTxDir = VSCSIIOREQTXDIR_READ;
153 uLbaStart = ((uint64_t) pVScsiReq->pbCDB[3]
154 | (pVScsiReq->pbCDB[2] << 8)
155 | ((pVScsiReq->pbCDB[1] & 0x1f) << 16));
156 cSectorTransfer = pVScsiReq->pbCDB[4];
157 break;
158 }
159 case SCSI_READ_10:
160 {
161 enmTxDir = VSCSIIOREQTXDIR_READ;
162 uLbaStart = vscsiBE2HU32(&pVScsiReq->pbCDB[2]);
163 cSectorTransfer = vscsiBE2HU16(&pVScsiReq->pbCDB[7]);
164 break;
165 }
166 case SCSI_READ_12:
167 {
168 enmTxDir = VSCSIIOREQTXDIR_READ;
169 uLbaStart = vscsiBE2HU32(&pVScsiReq->pbCDB[2]);
170 cSectorTransfer = vscsiBE2HU32(&pVScsiReq->pbCDB[6]);
171 break;
172 }
173 case SCSI_READ_16:
174 {
175 enmTxDir = VSCSIIOREQTXDIR_READ;
176 uLbaStart = vscsiBE2HU64(&pVScsiReq->pbCDB[2]);
177 cSectorTransfer = vscsiBE2HU32(&pVScsiReq->pbCDB[10]);
178 break;
179 }
180 case SCSI_READ_BUFFER:
181 {
182 uint8_t uDataMode = pVScsiReq->pbCDB[1] & 0x1f;
183
184 switch (uDataMode)
185 {
186 case 0x00:
187 case 0x01:
188 case 0x02:
189 case 0x03:
190 case 0x0a:
191 break;
192 case 0x0b:
193 {
194 uint8_t aReply[4];
195
196 /* We do not implement an echo buffer. */
197 memset(aReply, 0, sizeof(aReply));
198
199 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
200 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
201 break;
202 }
203 case 0x1a:
204 case 0x1c:
205 break;
206 default:
207 AssertMsgFailed(("Invalid data mode\n"));
208 }
209 break;
210 }
211 case SCSI_VERIFY_10:
212 case SCSI_START_STOP_UNIT:
213 {
214 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
215 break;
216 }
217 case SCSI_LOG_SENSE:
218 {
219 uint16_t cbMax = vscsiBE2HU16(&pVScsiReq->pbCDB[7]);
220 uint8_t uPageCode = pVScsiReq->pbCDB[2] & 0x3f;
221 uint8_t uSubPageCode = pVScsiReq->pbCDB[3];
222
223 switch (uPageCode)
224 {
225 case 0x00:
226 {
227 if (uSubPageCode == 0)
228 {
229 uint8_t aReply[4];
230
231 aReply[0] = 0;
232 aReply[1] = 0;
233 aReply[2] = 0;
234 aReply[3] = 0;
235
236 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
237 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
238 break;
239 }
240 }
241 default:
242 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00);
243 }
244 break;
245 }
246 case SCSI_SERVICE_ACTION_IN_16:
247 {
248 switch (pVScsiReq->pbCDB[1] & 0x1f)
249 {
250 case SCSI_SVC_ACTION_IN_READ_CAPACITY_16:
251 {
252 uint8_t aReply[32];
253
254 memset(aReply, 0, sizeof(aReply));
255 vscsiH2BEU64(aReply, pVScsiLunMmc->cSectors - 1);
256 vscsiH2BEU32(&aReply[8], pVScsiLunMmc->cbSector);
257 /* Leave the rest 0 */
258 RTSgBufCopyFromBuf(&pVScsiReq->SgBuf, aReply, sizeof(aReply));
259 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
260 break;
261 }
262 default:
263 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_INV_FIELD_IN_CMD_PACKET, 0x00); /* Don't know if this is correct */
264 }
265 break;
266 }
267 case SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL:
268 {
269 pVScsiLunMmc->fLocked = pVScsiReq->pbCDB[4] & 1;
270 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
271 break;
272 }
273
274 default:
275 //AssertMsgFailed(("Command %#x [%s] not implemented\n", pRequest->pbCDB[0], SCSICmdText(pRequest->pbCDB[0])));
276 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_ILLEGAL_OPCODE, 0x00);
277 }
278
279 if (enmTxDir != VSCSIIOREQTXDIR_INVALID)
280 {
281 LogFlow(("%s: uLbaStart=%llu cSectorTransfer=%u\n",
282 __FUNCTION__, uLbaStart, cSectorTransfer));
283
284 if (RT_UNLIKELY(uLbaStart + cSectorTransfer > pVScsiLunMmc->cSectors))
285 {
286 rcReq = vscsiLunReqSenseErrorSet(pVScsiLun, pVScsiReq, SCSI_SENSE_ILLEGAL_REQUEST, SCSI_ASC_LOGICAL_BLOCK_OOR, 0x00);
287 vscsiDeviceReqComplete(pVScsiLun->pVScsiDevice, pVScsiReq, rcReq, false, VINF_SUCCESS);
288 }
289 else if (!cSectorTransfer)
290 {
291 /* A 0 transfer length is not an error. */
292 rcReq = vscsiLunReqSenseOkSet(pVScsiLun, pVScsiReq);
293 vscsiDeviceReqComplete(pVScsiLun->pVScsiDevice, pVScsiReq, rcReq, false, VINF_SUCCESS);
294 }
295 else
296 {
297 /* Enqueue new I/O request */
298 rc = vscsiIoReqTransferEnqueue(pVScsiLun, pVScsiReq, enmTxDir,
299 uLbaStart * pVScsiLunMmc->cbSector,
300 cSectorTransfer * pVScsiLunMmc->cbSector);
301 }
302 }
303 else /* Request completed */
304 vscsiDeviceReqComplete(pVScsiLun->pVScsiDevice, pVScsiReq, rcReq, false, VINF_SUCCESS);
305
306 return rc;
307}
308
309VSCSILUNDESC g_VScsiLunTypeMmc =
310{
311 /** enmLunType */
312 VSCSILUNTYPE_MMC,
313 /** pcszDescName */
314 "MMC",
315 /** cbLun */
316 sizeof(VSCSILUNMMC),
317 /** pfnVScsiLunInit */
318 vscsiLunMmcInit,
319 /** pfnVScsiLunDestroy */
320 vscsiLunMmcDestroy,
321 /** pfnVScsiLunReqProcess */
322 vscsiLunMmcReqProcess
323};
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