VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvHostBase-freebsd.cpp@ 76734

Last change on this file since 76734 was 76553, checked in by vboxsync, 6 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.2 KB
Line 
1/* $Id: DrvHostBase-freebsd.cpp 76553 2019-01-01 01:45:53Z vboxsync $ */
2/** @file
3 * DrvHostBase - Host base drive access driver, FreeBSD specifics.
4 */
5
6/*
7 * Copyright (C) 2006-2019 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_BASE
23#include <sys/cdefs.h>
24#include <sys/param.h>
25#include <errno.h>
26#include <stdio.h>
27#include <cam/cam.h>
28#include <cam/cam_ccb.h>
29#include <cam/scsi/scsi_message.h>
30#include <cam/scsi/scsi_pass.h>
31#include <VBox/err.h>
32
33#include <VBox/scsi.h>
34#include <iprt/log.h>
35
36
37/*********************************************************************************************************************************
38* Structures and Typedefs *
39*********************************************************************************************************************************/
40/**
41 * Host backend specific data.
42 */
43typedef struct DRVHOSTBASEOS
44{
45 /** The filehandle of the device. */
46 RTFILE hFileDevice;
47 /** The block size. Set when querying the media size. */
48 uint32_t cbBlock;
49 /** SCSI bus number. */
50 path_id_t ScsiBus;
51 /** target ID of the passthrough device. */
52 target_id_t ScsiTargetID;
53 /** LUN of the passthrough device. */
54 lun_id_t ScsiLunID;
55} DRVHOSTBASEOS;
56/** Pointer to the host backend specific data. */
57typedef DRVHOSTBASEOS *PDRVHOSBASEOS;
58AssertCompile(sizeof(DRVHOSTBASEOS) <= 64);
59
60#define DRVHOSTBASE_OS_INT_DECLARED
61#include "DrvHostBase.h"
62
63
64/*********************************************************************************************************************************
65* Defined Constants And Macros *
66*********************************************************************************************************************************/
67/** Maximum buffer size supported by the CAM subsystem. */
68#define FBSD_SCSI_MAX_BUFFER_SIZE (64 * _1K)
69
70
71
72DECLHIDDEN(int) drvHostBaseScsiCmdOs(PDRVHOSTBASE pThis, const uint8_t *pbCmd, size_t cbCmd, PDMMEDIATXDIR enmTxDir,
73 void *pvBuf, uint32_t *pcbBuf, uint8_t *pbSense, size_t cbSense, uint32_t cTimeoutMillies)
74{
75 /*
76 * Minimal input validation.
77 */
78 Assert(enmTxDir == PDMMEDIATXDIR_NONE || enmTxDir == PDMMEDIATXDIR_FROM_DEVICE || enmTxDir == PDMMEDIATXDIR_TO_DEVICE);
79 Assert(!pvBuf || pcbBuf);
80 Assert(pvBuf || enmTxDir == PDMMEDIATXDIR_NONE);
81 Assert(pbSense || !cbSense);
82 AssertPtr(pbCmd);
83 Assert(cbCmd <= 16 && cbCmd >= 1);
84 const uint32_t cbBuf = pcbBuf ? *pcbBuf : 0;
85 if (pcbBuf)
86 *pcbBuf = 0;
87
88 int rc = VINF_SUCCESS;
89 int rcBSD = 0;
90 union ccb DeviceCCB;
91 union ccb *pDeviceCCB = &DeviceCCB;
92 u_int32_t fFlags;
93
94 memset(pDeviceCCB, 0, sizeof(DeviceCCB));
95 pDeviceCCB->ccb_h.path_id = pThis->Os.ScsiBus;
96 pDeviceCCB->ccb_h.target_id = pThis->Os.ScsiTargetID;
97 pDeviceCCB->ccb_h.target_lun = pThis->Os.ScsiLunID;
98
99 /* The SCSI INQUIRY command can't be passed through directly. */
100 if (pbCmd[0] == SCSI_INQUIRY)
101 {
102 pDeviceCCB->ccb_h.func_code = XPT_GDEV_TYPE;
103
104 rcBSD = ioctl(RTFileToNative(pThis->Os.hFileDevice), CAMIOCOMMAND, pDeviceCCB);
105 if (!rcBSD)
106 {
107 uint32_t cbCopy = cbBuf < sizeof(struct scsi_inquiry_data)
108 ? cbBuf
109 : sizeof(struct scsi_inquiry_data);;
110 memcpy(pvBuf, &pDeviceCCB->cgd.inq_data, cbCopy);
111 memset(pbSense, 0, cbSense);
112
113 if (pcbBuf)
114 *pcbBuf = cbCopy;
115 }
116 else
117 rc = RTErrConvertFromErrno(errno);
118 }
119 else
120 {
121 /* Copy the CDB. */
122 memcpy(&pDeviceCCB->csio.cdb_io.cdb_bytes, pbCmd, cbCmd);
123
124 /* Set direction. */
125 if (enmTxDir == PDMMEDIATXDIR_NONE)
126 fFlags = CAM_DIR_NONE;
127 else if (enmTxDir == PDMMEDIATXDIR_FROM_DEVICE)
128 fFlags = CAM_DIR_IN;
129 else
130 fFlags = CAM_DIR_OUT;
131
132 fFlags |= CAM_DEV_QFRZDIS;
133
134 cam_fill_csio(&pDeviceCCB->csio, 1, NULL, fFlags, MSG_SIMPLE_Q_TAG,
135 (u_int8_t *)pvBuf, cbBuf, cbSense, cbCmd,
136 cTimeoutMillies ? cTimeoutMillies : 30000/* timeout */);
137
138 /* Send command */
139 rcBSD = ioctl(RTFileToNative(pThis->Os.hFileDevice), CAMIOCOMMAND, pDeviceCCB);
140 if (!rcBSD)
141 {
142 switch (pDeviceCCB->ccb_h.status & CAM_STATUS_MASK)
143 {
144 case CAM_REQ_CMP:
145 rc = VINF_SUCCESS;
146 break;
147 case CAM_SEL_TIMEOUT:
148 rc = VERR_DEV_IO_ERROR;
149 break;
150 case CAM_CMD_TIMEOUT:
151 rc = VERR_TIMEOUT;
152 break;
153 default:
154 rc = VERR_DEV_IO_ERROR;
155 }
156
157 if (pcbBuf)
158 *pcbBuf = cbBuf - pDeviceCCB->csio.resid;
159
160 if (pbSense)
161 memcpy(pbSense, &pDeviceCCB->csio.sense_data,
162 cbSense - pDeviceCCB->csio.sense_resid);
163 }
164 else
165 rc = RTErrConvertFromErrno(errno);
166 }
167}
168
169
170DECLHIDDEN(size_t) drvHostBaseScsiCmdGetBufLimitOs(PDRVHOSTBASE pThis)
171{
172 RT_NOREF(pThis);
173
174 return FBSD_SCSI_MAX_BUFFER_SIZE;
175}
176
177
178DECLHIDDEN(int) drvHostBaseGetMediaSizeOs(PDRVHOSTBASE pThis, uint64_t *pcb)
179{
180 /*
181 * Try a READ_CAPACITY command...
182 */
183 struct
184 {
185 uint32_t cBlocks;
186 uint32_t cbBlock;
187 } Buf = {0, 0};
188 uint32_t cbBuf = sizeof(Buf);
189 uint8_t abCmd[16] =
190 {
191 SCSI_READ_CAPACITY, 0, 0, 0, 0, 0, 0,
192 0,0,0,0,0,0,0,0,0
193 };
194 int rc = drvHostBaseScsiCmdOs(pThis, abCmd, 6, PDMMEDIATXDIR_FROM_DEVICE, &Buf, &cbBuf, NULL, 0, 0);
195 if (RT_SUCCESS(rc))
196 {
197 Assert(cbBuf == sizeof(Buf));
198 Buf.cBlocks = RT_BE2H_U32(Buf.cBlocks);
199 Buf.cbBlock = RT_BE2H_U32(Buf.cbBlock);
200 //if (Buf.cbBlock > 2048) /* everyone else is doing this... check if it needed/right.*/
201 // Buf.cbBlock = 2048;
202 pThis->Os.cbBlock = Buf.cbBlock;
203
204 *pcb = (uint64_t)Buf.cBlocks * Buf.cbBlock;
205 }
206 return rc;
207}
208
209
210DECLHIDDEN(int) drvHostBaseReadOs(PDRVHOSTBASE pThis, uint64_t off, void *pvBuf, size_t cbRead)
211{
212 int rc = VINF_SUCCESS;
213
214 if (pThis->Os.cbBlock)
215 {
216 /*
217 * Issue a READ(12) request.
218 */
219 do
220 {
221 const uint32_t LBA = off / pThis->Os.cbBlock;
222 AssertReturn(!(off % pThis->Os.cbBlock), VERR_INVALID_PARAMETER);
223 uint32_t cbRead32 = cbRead > FBSD_SCSI_MAX_BUFFER_SIZE
224 ? FBSD_SCSI_MAX_BUFFER_SIZE
225 : (uint32_t)cbRead;
226 const uint32_t cBlocks = cbRead32 / pThis->Os.cbBlock;
227 AssertReturn(!(cbRead % pThis->Os.cbBlock), VERR_INVALID_PARAMETER);
228 uint8_t abCmd[16] =
229 {
230 SCSI_READ_12, 0,
231 RT_BYTE4(LBA), RT_BYTE3(LBA), RT_BYTE2(LBA), RT_BYTE1(LBA),
232 RT_BYTE4(cBlocks), RT_BYTE3(cBlocks), RT_BYTE2(cBlocks), RT_BYTE1(cBlocks),
233 0, 0, 0, 0, 0
234 };
235 rc = drvHostBaseScsiCmdOs(pThis, abCmd, 12, PDMMEDIATXDIR_FROM_DEVICE, pvBuf, &cbRead32, NULL, 0, 0);
236
237 off += cbRead32;
238 cbRead -= cbRead32;
239 pvBuf = (uint8_t *)pvBuf + cbRead32;
240 } while ((cbRead > 0) && RT_SUCCESS(rc));
241 }
242 else
243 rc = VERR_MEDIA_NOT_PRESENT;
244
245 return rc;
246}
247
248
249DECLHIDDEN(int) drvHostBaseWriteOs(PDRVHOSTBASE pThis, uint64_t off, const void *pvBuf, size_t cbWrite)
250{
251 RT_NOREF4(pThis, off, pvBuf, cbWrite);
252 return VERR_WRITE_PROTECT;
253}
254
255
256DECLHIDDEN(int) drvHostBaseFlushOs(PDRVHOSTBASE pThis)
257{
258 RT_NOREF1(pThis);
259 return VINF_SUCCESS;
260}
261
262
263DECLHIDDEN(int) drvHostBaseDoLockOs(PDRVHOSTBASE pThis, bool fLock)
264{
265 uint8_t abCmd[16] =
266 {
267 SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL, 0, 0, 0, fLock, 0,
268 0,0,0,0,0,0,0,0,0,0
269 };
270 return drvHostBaseScsiCmdOs(pThis, abCmd, 6, PDMMEDIATXDIR_NONE, NULL, NULL, NULL, 0, 0);
271}
272
273
274DECLHIDDEN(int) drvHostBaseEjectOs(PDRVHOSTBASE pThis)
275{
276 uint8_t abCmd[16] =
277 {
278 SCSI_START_STOP_UNIT, 0, 0, 0, 2 /*eject+stop*/, 0,
279 0,0,0,0,0,0,0,0,0,0
280 };
281 return drvHostBaseScsiCmdOs(pThis, abCmd, 6, PDMMEDIATXDIR_NONE, NULL, NULL, NULL, 0, 0);
282}
283
284
285DECLHIDDEN(int) drvHostBaseQueryMediaStatusOs(PDRVHOSTBASE pThis, bool *pfMediaChanged, bool *pfMediaPresent)
286{
287 /*
288 * Issue a TEST UNIT READY request.
289 */
290 *pfMediaChanged = false;
291 *pfMediaPresent = false;
292 uint8_t abCmd[16] = { SCSI_TEST_UNIT_READY, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 };
293 uint8_t abSense[32];
294 int rc = drvHostBaseScsiCmdOs(pThis, abCmd, 6, PDMMEDIATXDIR_NONE, NULL, NULL, abSense, sizeof(abSense), 0);
295 if (RT_SUCCESS(rc))
296 *pfMediaPresent = true;
297 else if ( rc == VERR_UNRESOLVED_ERROR
298 && abSense[2] == 6 /* unit attention */
299 && ( (abSense[12] == 0x29 && abSense[13] < 5 /* reset */)
300 || (abSense[12] == 0x2a && abSense[13] == 0 /* parameters changed */) //???
301 || (abSense[12] == 0x3f && abSense[13] == 0 /* target operating conditions have changed */) //???
302 || (abSense[12] == 0x3f && abSense[13] == 2 /* changed operating definition */) //???
303 || (abSense[12] == 0x3f && abSense[13] == 3 /* inquiry parameters changed */)
304 || (abSense[12] == 0x3f && abSense[13] == 5 /* device identifier changed */)
305 )
306 )
307 {
308 *pfMediaPresent = false;
309 *pfMediaChanged = true;
310 rc = VINF_SUCCESS;
311 /** @todo check this media change stuff on Darwin. */
312 }
313
314 return rc;
315}
316
317
318DECLHIDDEN(void) drvHostBaseInitOs(PDRVHOSTBASE pThis)
319{
320 pThis->Os.hFileDevice = NIL_RTFILE;
321}
322
323
324DECLHIDDEN(int) drvHostBaseOpenOs(PDRVHOSTBASE pThis, bool fReadOnly)
325{
326 RT_NOREF(fReadOnly);
327 RTFILE hFileDevice;
328 int rc = RTFileOpen(&hFileDevice, pThis->pszDevice, RTFILE_O_READWRITE | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
329 if (RT_FAILURE(rc))
330 return rc;
331
332 /*
333 * The current device handle can't passthrough SCSI commands.
334 * We have to get he passthrough device path and open this.
335 */
336 union ccb DeviceCCB;
337 memset(&DeviceCCB, 0, sizeof(DeviceCCB));
338
339 DeviceCCB.ccb_h.func_code = XPT_GDEVLIST;
340 int rcBSD = ioctl(RTFileToNative(hFileDevice), CAMGETPASSTHRU, &DeviceCCB);
341 if (!rcBSD)
342 {
343 char *pszPassthroughDevice = NULL;
344 rc = RTStrAPrintf(&pszPassthroughDevice, "/dev/%s%u",
345 DeviceCCB.cgdl.periph_name, DeviceCCB.cgdl.unit_number);
346 if (rc >= 0)
347 {
348 RTFILE hPassthroughDevice;
349 rc = RTFileOpen(&hPassthroughDevice, pszPassthroughDevice, RTFILE_O_READWRITE | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
350 RTStrFree(pszPassthroughDevice);
351 if (RT_SUCCESS(rc))
352 {
353 /* Get needed device parameters. */
354
355 /*
356 * The device path, target id and lun id. Those are
357 * needed for the SCSI passthrough ioctl.
358 */
359 memset(&DeviceCCB, 0, sizeof(DeviceCCB));
360 DeviceCCB.ccb_h.func_code = XPT_GDEVLIST;
361
362 rcBSD = ioctl(RTFileToNative(hPassthroughDevice), CAMGETPASSTHRU, &DeviceCCB);
363 if (!rcBSD)
364 {
365 if (DeviceCCB.cgdl.status != CAM_GDEVLIST_ERROR)
366 {
367 pThis->Os.ScsiBus = DeviceCCB.ccb_h.path_id;
368 pThis->Os.ScsiTargetID = DeviceCCB.ccb_h.target_id;
369 pThis->Os.ScsiLunID = DeviceCCB.ccb_h.target_lun;
370 pThis->Os.hFileDevice = hPassthroughDevice;
371 }
372 else
373 {
374 /* The passthrough device wasn't found. */
375 rc = VERR_NOT_FOUND;
376 }
377 }
378 else
379 rc = RTErrConvertFromErrno(errno);
380
381 if (RT_FAILURE(rc))
382 RTFileClose(hPassthroughDevice);
383 }
384 }
385 else
386 rc = VERR_NO_STR_MEMORY;
387 }
388 else
389 rc = RTErrConvertFromErrno(errno);
390
391 RTFileClose(hFileDevice);
392 return rc;
393}
394
395
396DECLHIDDEN(int) drvHostBaseMediaRefreshOs(PDRVHOSTBASE pThis)
397{
398 RT_NOREF(pThis);
399 return VINF_SUCCESS;
400}
401
402
403DECLHIDDEN(bool) drvHostBaseIsMediaPollingRequiredOs(PDRVHOSTBASE pThis)
404{
405 if (pThis->enmType == PDMMEDIATYPE_CDROM || pThis->enmType == PDMMEDIATYPE_DVD)
406 return true;
407
408 AssertMsgFailed(("FreeBSD supports only CD/DVD host drive access\n"));
409 return false;
410}
411
412
413DECLHIDDEN(void) drvHostBaseDestructOs(PDRVHOSTBASE pThis)
414{
415 /*
416 * Unlock the drive if we've locked it or we're in passthru mode.
417 */
418 if ( pThis->fLocked
419 && pThis->Os.hFileDevice != NIL_RTFILE
420 && pThis->pfnDoLock)
421 {
422 int rc = pThis->pfnDoLock(pThis, false);
423 if (RT_SUCCESS(rc))
424 pThis->fLocked = false;
425 }
426
427 if (pThis->Os.hFileDevice != NIL_RTFILE)
428 {
429 int rc = RTFileClose(pThis->Os.hFileDevice);
430 AssertRC(rc);
431 pThis->Os.hFileDevice = NIL_RTFILE;
432 }
433}
434
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