VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvHostBase-linux.cpp@ 64700

Last change on this file since 64700 was 64316, checked in by vboxsync, 8 years ago

Devices/Storage/DrvHost*: Move host dependent members of DRVHOSTBASE into a private struct for each host to keep including host dependent headers in one file for each host

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 11.6 KB
Line 
1/* $Id: DrvHostBase-linux.cpp 64316 2016-10-19 11:59:42Z vboxsync $ */
2/** @file
3 * DrvHostBase - Host base drive access driver, Linux specifics.
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#define LOG_GROUP LOG_GROUP_DRV_HOST_BASE
18#include <sys/ioctl.h>
19#include <sys/fcntl.h>
20#include <errno.h>
21#include <linux/version.h>
22/* All the following crap is apparently not necessary anymore since Linux
23 * version 2.6.29. */
24#if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 29)
25/* This is a hack to work around conflicts between these linux kernel headers
26 * and the GLIBC tcpip headers. They have different declarations of the 4
27 * standard byte order functions. */
28# define _LINUX_BYTEORDER_GENERIC_H
29/* This is another hack for not bothering with C++ unfriendly byteswap macros. */
30/* Those macros that are needed are defined in the header below. */
31# include "swab.h"
32#endif
33#include <linux/fd.h>
34#include <linux/cdrom.h>
35#include <limits.h>
36
37#include <iprt/mem.h>
38#include <iprt/file.h>
39#include <VBox/scsi.h>
40
41/**
42 * Host backend specific data.
43 */
44typedef struct DRVHOSTBASEOS
45{
46 /** The filehandle of the device. */
47 RTFILE hFileDevice;
48 /** Double buffer required for ioctl with the Linux kernel as long as we use
49 * remap_pfn_range() instead of vm_insert_page(). */
50 uint8_t *pbDoubleBuffer;
51 /** Previous disk inserted indicator for the media polling on floppy drives. */
52 bool fPrevDiskIn;
53} DRVHOSTBASEOS;
54/** Pointer to the host backend specific data. */
55typedef DRVHOSTBASEOS *PDRVHOSBASEOS;
56AssertCompile(sizeof(DRVHOSTBASEOS) <= 64);
57
58#define DRVHOSTBASE_OS_INT_DECLARED
59#include "DrvHostBase.h"
60
61DECLHIDDEN(int) drvHostBaseScsiCmdOs(PDRVHOSTBASE pThis, const uint8_t *pbCmd, size_t cbCmd, PDMMEDIATXDIR enmTxDir,
62 void *pvBuf, uint32_t *pcbBuf, uint8_t *pbSense, size_t cbSense, uint32_t cTimeoutMillies)
63{
64 /*
65 * Minimal input validation.
66 */
67 Assert(enmTxDir == PDMMEDIATXDIR_NONE || enmTxDir == PDMMEDIATXDIR_FROM_DEVICE || enmTxDir == PDMMEDIATXDIR_TO_DEVICE);
68 Assert(!pvBuf || pcbBuf);
69 Assert(pvBuf || enmTxDir == PDMMEDIATXDIR_NONE);
70 Assert(pbSense || !cbSense); RT_NOREF(cbSense);
71 AssertPtr(pbCmd);
72 Assert(cbCmd <= 16 && cbCmd >= 1);
73
74 /* Allocate the temporary buffer lazily. */
75 if(RT_UNLIKELY(!pThis->Os.pbDoubleBuffer))
76 {
77 pThis->Os.pbDoubleBuffer = (uint8_t *)RTMemAlloc(SCSI_MAX_BUFFER_SIZE);
78 if (!pThis->Os.pbDoubleBuffer)
79 return VERR_NO_MEMORY;
80 }
81
82 int rc = VERR_GENERAL_FAILURE;
83 int direction;
84 struct cdrom_generic_command cgc;
85
86 switch (enmTxDir)
87 {
88 case PDMMEDIATXDIR_NONE:
89 Assert(*pcbBuf == 0);
90 direction = CGC_DATA_NONE;
91 break;
92 case PDMMEDIATXDIR_FROM_DEVICE:
93 Assert(*pcbBuf != 0);
94 Assert(*pcbBuf <= SCSI_MAX_BUFFER_SIZE);
95 /* Make sure that the buffer is clear for commands reading
96 * data. The actually received data may be shorter than what
97 * we expect, and due to the unreliable feedback about how much
98 * data the ioctl actually transferred, it's impossible to
99 * prevent that. Returning previous buffer contents may cause
100 * security problems inside the guest OS, if users can issue
101 * commands to the CDROM device. */
102 memset(pThis->Os.pbDoubleBuffer, '\0', *pcbBuf);
103 direction = CGC_DATA_READ;
104 break;
105 case PDMMEDIATXDIR_TO_DEVICE:
106 Assert(*pcbBuf != 0);
107 Assert(*pcbBuf <= SCSI_MAX_BUFFER_SIZE);
108 memcpy(pThis->Os.pbDoubleBuffer, pvBuf, *pcbBuf);
109 direction = CGC_DATA_WRITE;
110 break;
111 default:
112 AssertMsgFailed(("enmTxDir invalid!\n"));
113 direction = CGC_DATA_NONE;
114 }
115 memset(&cgc, '\0', sizeof(cgc));
116 memcpy(cgc.cmd, pbCmd, RT_MIN(CDROM_PACKET_SIZE, cbCmd));
117 cgc.buffer = (unsigned char *)pThis->Os.pbDoubleBuffer;
118 cgc.buflen = *pcbBuf;
119 cgc.stat = 0;
120 Assert(cbSense >= sizeof(struct request_sense));
121 cgc.sense = (struct request_sense *)pbSense;
122 cgc.data_direction = direction;
123 cgc.quiet = false;
124 cgc.timeout = cTimeoutMillies;
125 rc = ioctl(RTFileToNative(pThis->Os.hFileDevice), CDROM_SEND_PACKET, &cgc);
126 if (rc < 0)
127 {
128 if (errno == EBUSY)
129 rc = VERR_PDM_MEDIA_LOCKED;
130 else if (errno == ENOSYS)
131 rc = VERR_NOT_SUPPORTED;
132 else
133 {
134 rc = RTErrConvertFromErrno(errno);
135 if (rc == VERR_ACCESS_DENIED && cgc.sense->sense_key == SCSI_SENSE_NONE)
136 cgc.sense->sense_key = SCSI_SENSE_ILLEGAL_REQUEST;
137 Log2(("%s: error status %d, rc=%Rrc\n", __FUNCTION__, cgc.stat, rc));
138 }
139 }
140 switch (enmTxDir)
141 {
142 case PDMMEDIATXDIR_FROM_DEVICE:
143 memcpy(pvBuf, pThis->Os.pbDoubleBuffer, *pcbBuf);
144 break;
145 default:
146 ;
147 }
148 Log2(("%s: after ioctl: cgc.buflen=%d txlen=%d\n", __FUNCTION__, cgc.buflen, *pcbBuf));
149 /* The value of cgc.buflen does not reliably reflect the actual amount
150 * of data transferred (for packet commands with little data transfer
151 * it's 0). So just assume that everything worked ok. */
152
153 return rc;
154}
155
156DECLHIDDEN(int) drvHostBaseGetMediaSizeOs(PDRVHOSTBASE pThis, uint64_t *pcb)
157{
158 int rc = VERR_INVALID_STATE;
159
160 if (PDMMEDIATYPE_IS_FLOPPY(pThis->enmType))
161 {
162 rc = ioctl(RTFileToNative(pThis->Os.hFileDevice), FDFLUSH);
163 if (rc)
164 {
165 rc = RTErrConvertFromErrno (errno);
166 Log(("DrvHostFloppy: FDFLUSH ioctl(%s) failed, errno=%d rc=%Rrc\n", pThis->pszDevice, errno, rc));
167 return rc;
168 }
169
170 floppy_drive_struct DrvStat;
171 rc = ioctl(RTFileToNative(pThis->Os.hFileDevice), FDGETDRVSTAT, &DrvStat);
172 if (rc)
173 {
174 rc = RTErrConvertFromErrno(errno);
175 Log(("DrvHostFloppy: FDGETDRVSTAT ioctl(%s) failed, errno=%d rc=%Rrc\n", pThis->pszDevice, errno, rc));
176 return rc;
177 }
178 pThis->fReadOnly = !(DrvStat.flags & FD_DISK_WRITABLE);
179 rc = RTFileSeek(pThis->Os.hFileDevice, 0, RTFILE_SEEK_END, pcb);
180 }
181 else if (pThis->enmType == PDMMEDIATYPE_CDROM || pThis->enmType == PDMMEDIATYPE_DVD)
182 {
183 /* Clear the media-changed-since-last-call-thingy just to be on the safe side. */
184 ioctl(RTFileToNative(pThis->Os.hFileDevice), CDROM_MEDIA_CHANGED, CDSL_CURRENT);
185 rc = RTFileSeek(pThis->Os.hFileDevice, 0, RTFILE_SEEK_END, pcb);
186 }
187
188 return rc;
189}
190
191
192DECLHIDDEN(int) drvHostBaseReadOs(PDRVHOSTBASE pThis, uint64_t off, void *pvBuf, size_t cbRead)
193{
194 return RTFileReadAt(pThis->Os.hFileDevice, off, pvBuf, cbRead, NULL);
195}
196
197
198DECLHIDDEN(int) drvHostBaseWriteOs(PDRVHOSTBASE pThis, uint64_t off, const void *pvBuf, size_t cbWrite)
199{
200 return RTFileWriteAt(pThis->Os.hFileDevice, off, pvBuf, cbWrite, NULL);
201}
202
203
204DECLHIDDEN(int) drvHostBaseFlushOs(PDRVHOSTBASE pThis)
205{
206 return RTFileFlush(pThis->Os.hFileDevice);
207}
208
209
210DECLHIDDEN(int) drvHostBaseDoLockOs(PDRVHOSTBASE pThis, bool fLock)
211{
212 int rc = ioctl(RTFileToNative(pThis->Os.hFileDevice), CDROM_LOCKDOOR, (int)fLock);
213 if (rc < 0)
214 {
215 if (errno == EBUSY)
216 rc = VERR_ACCESS_DENIED;
217 else if (errno == EDRIVE_CANT_DO_THIS)
218 rc = VERR_NOT_SUPPORTED;
219 else
220 rc = RTErrConvertFromErrno(errno);
221 }
222
223 return rc;
224}
225
226
227DECLHIDDEN(int) drvHostBaseEjectOs(PDRVHOSTBASE pThis)
228{
229 int rc = ioctl(RTFileToNative(pThis->Os.hFileDevice), CDROMEJECT, 0);
230 if (rc < 0)
231 {
232 if (errno == EBUSY)
233 rc = VERR_PDM_MEDIA_LOCKED;
234 else if (errno == ENOSYS)
235 rc = VERR_NOT_SUPPORTED;
236 else
237 rc = RTErrConvertFromErrno(errno);
238 }
239
240 return rc;
241}
242
243
244DECLHIDDEN(int) drvHostBaseQueryMediaStatusOs(PDRVHOSTBASE pThis, bool *pfMediaChanged, bool *pfMediaPresent)
245{
246 int rc = VINF_SUCCESS;
247
248 if (PDMMEDIATYPE_IS_FLOPPY(pThis->enmType))
249 {
250 floppy_drive_struct DrvStat;
251 int rcLnx = ioctl(RTFileToNative(pThis->Os.hFileDevice), FDPOLLDRVSTAT, &DrvStat);
252 if (!rcLnx)
253 {
254 bool fDiskIn = !(DrvStat.flags & (FD_VERIFY | FD_DISK_NEWCHANGE));
255 *pfMediaPresent = fDiskIn;
256
257 if (fDiskIn != pThis->Os.fPrevDiskIn)
258 *pfMediaChanged = true;
259
260 pThis->Os.fPrevDiskIn = fDiskIn;
261 }
262 else
263 rc = RTErrConvertFromErrno(errno);
264 }
265 else
266 {
267 *pfMediaPresent = ioctl(RTFileToNative(pThis->Os.hFileDevice), CDROM_DRIVE_STATUS, CDSL_CURRENT) == CDS_DISC_OK;
268 *pfMediaChanged = false;
269 if (pThis->fMediaPresent != *pfMediaPresent)
270 *pfMediaChanged = ioctl(RTFileToNative(pThis->Os.hFileDevice), CDROM_MEDIA_CHANGED, CDSL_CURRENT) == 1;
271 }
272
273 return rc;
274}
275
276
277DECLHIDDEN(void) drvHostBaseInitOs(PDRVHOSTBASE pThis)
278{
279 pThis->Os.hFileDevice = NIL_RTFILE;
280 pThis->Os.pbDoubleBuffer = NULL;
281 pThis->Os.fPrevDiskIn = false;
282}
283
284
285DECLHIDDEN(int) drvHostBaseOpenOs(PDRVHOSTBASE pThis, bool fReadOnly)
286{
287 uint32_t fFlags = (fReadOnly ? RTFILE_O_READ : RTFILE_O_READWRITE) | RTFILE_O_OPEN | RTFILE_O_DENY_NONE | RTFILE_O_NON_BLOCK;
288 return RTFileOpen(&pThis->Os.hFileDevice, pThis->pszDevice, fFlags);
289}
290
291
292DECLHIDDEN(int) drvHostBaseMediaRefreshOs(PDRVHOSTBASE pThis)
293{
294 /*
295 * Need to re-open the device because it will kill off any cached data
296 * that Linux for some peculiar reason thinks should survive a media change.
297 */
298 if (pThis->Os.hFileDevice != NIL_RTFILE)
299 {
300 RTFileClose(pThis->Os.hFileDevice);
301 pThis->Os.hFileDevice = NIL_RTFILE;
302 }
303 int rc = drvHostBaseOpenOs(pThis, pThis->fReadOnlyConfig);
304 if (RT_FAILURE(rc))
305 {
306 if (!pThis->fReadOnlyConfig)
307 {
308 LogFlow(("%s-%d: drvHostBaseMediaRefreshOs: '%s' - retry readonly (%Rrc)\n",
309 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDevice, rc));
310 rc = drvHostBaseOpenOs(pThis, true);
311 }
312 if (RT_FAILURE(rc))
313 {
314 LogFlow(("%s-%d: failed to open device '%s', rc=%Rrc\n",
315 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDevice, rc));
316 return rc;
317 }
318 pThis->fReadOnly = true;
319 }
320 else
321 pThis->fReadOnly = pThis->fReadOnlyConfig;
322
323 return rc;
324}
325
326
327DECLHIDDEN(bool) drvHostBaseIsMediaPollingRequiredOs(PDRVHOSTBASE pThis)
328{
329 RT_NOREF(pThis);
330 return true; /* On Linux we always use media polling. */
331}
332
333
334DECLHIDDEN(void) drvHostBaseDestructOs(PDRVHOSTBASE pThis)
335{
336 /*
337 * Unlock the drive if we've locked it or we're in passthru mode.
338 */
339 if ( pThis->fLocked
340 && pThis->Os.hFileDevice != NIL_RTFILE
341 && pThis->pfnDoLock)
342 {
343 int rc = pThis->pfnDoLock(pThis, false);
344 if (RT_SUCCESS(rc))
345 pThis->fLocked = false;
346 }
347
348 if (pThis->Os.pbDoubleBuffer)
349 {
350 RTMemFree(pThis->Os.pbDoubleBuffer);
351 pThis->Os.pbDoubleBuffer = NULL;
352 }
353
354 if (pThis->Os.hFileDevice != NIL_RTFILE)
355 {
356 int rc = RTFileClose(pThis->Os.hFileDevice);
357 AssertRC(rc);
358 pThis->Os.hFileDevice = NIL_RTFILE;
359 }
360}
361
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