VirtualBox

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

Last change on this file since 27436 was 26173, checked in by vboxsync, 15 years ago

PDM: s/pCfgHandle/pCfg/g - part 2.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 26.6 KB
Line 
1/* $Id: DrvHostDVD.cpp 26173 2010-02-02 21:11:09Z vboxsync $ */
2/** @file
3 * DrvHostDVD - Host DVD block driver.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#define LOG_GROUP LOG_GROUP_DRV_HOST_DVD
27#define __STDC_LIMIT_MACROS
28#define __STDC_CONSTANT_MACROS
29#ifdef RT_OS_DARWIN
30# include <mach/mach.h>
31# include <Carbon/Carbon.h>
32# include <IOKit/IOKitLib.h>
33# include <IOKit/IOCFPlugIn.h>
34# include <IOKit/scsi/SCSITaskLib.h>
35# include <IOKit/scsi/SCSICommandOperationCodes.h>
36# include <IOKit/storage/IOStorageDeviceCharacteristics.h>
37# include <mach/mach_error.h>
38# define USE_MEDIA_POLLING
39
40#elif defined(RT_OS_L4)
41/* nothing (yet). */
42
43#elif defined RT_OS_LINUX
44# include <sys/ioctl.h>
45# include <linux/version.h>
46/* All the following crap is apparently not necessary anymore since Linux
47 * version 2.6.29. */
48# if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 29)
49/* This is a hack to work around conflicts between these linux kernel headers
50 * and the GLIBC tcpip headers. They have different declarations of the 4
51 * standard byte order functions. */
52# define _LINUX_BYTEORDER_GENERIC_H
53/* This is another hack for not bothering with C++ unfriendly byteswap macros. */
54/* Those macros that are needed are defined in the header below. */
55# include "swab.h"
56# endif
57# include <linux/cdrom.h>
58# include <sys/fcntl.h>
59# include <errno.h>
60# include <limits.h>
61# include <iprt/mem.h>
62# define USE_MEDIA_POLLING
63
64#elif defined(RT_OS_SOLARIS)
65# include <stropts.h>
66# include <fcntl.h>
67# include <errno.h>
68# include <pwd.h>
69# include <unistd.h>
70# include <syslog.h>
71# ifdef VBOX_WITH_SUID_WRAPPER
72# include <auth_attr.h>
73# endif
74# include <sys/dkio.h>
75# include <sys/sockio.h>
76# include <sys/scsi/scsi.h>
77# define USE_MEDIA_POLLING
78
79#elif defined(RT_OS_WINDOWS)
80# include <Windows.h>
81# include <winioctl.h>
82# include <ntddscsi.h>
83# undef USE_MEDIA_POLLING
84
85#elif defined(RT_OS_FREEBSD)
86# include <sys/cdefs.h>
87# include <sys/param.h>
88# include <stdio.h>
89# include <cam/cam.h>
90# include <cam/cam_ccb.h>
91# define USE_MEDIA_POLLING
92
93#else
94# error "Unsupported Platform."
95#endif
96
97#include <VBox/pdmdrv.h>
98#include <iprt/assert.h>
99#include <iprt/file.h>
100#include <iprt/string.h>
101#include <iprt/thread.h>
102#include <iprt/critsect.h>
103#include <VBox/scsi.h>
104
105#include "Builtins.h"
106#include "DrvHostBase.h"
107
108
109/* Forward declarations. */
110
111static DECLCALLBACK(int) drvHostDvdDoLock(PDRVHOSTBASE pThis, bool fLock);
112#ifdef VBOX_WITH_SUID_WRAPPER
113static int solarisCheckUserAuth();
114static int solarisEnterRootMode(uid_t *pEffUserID);
115static int solarisExitRootMode(uid_t *pEffUserID);
116#endif
117
118
119/** @copydoc PDMIMOUNT::pfnUnmount */
120static DECLCALLBACK(int) drvHostDvdUnmount(PPDMIMOUNT pInterface, bool fForce)
121{
122 PDRVHOSTBASE pThis = PDMIMOUNT_2_DRVHOSTBASE(pInterface);
123 RTCritSectEnter(&pThis->CritSect);
124
125 /*
126 * Validate state.
127 */
128 int rc = VINF_SUCCESS;
129 if (!pThis->fLocked || fForce)
130 {
131 /* Unlock drive if necessary. */
132 if (pThis->fLocked)
133 drvHostDvdDoLock(pThis, false);
134
135 /*
136 * Eject the disc.
137 */
138#if defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
139 uint8_t abCmd[16] =
140 {
141 SCSI_START_STOP_UNIT, 0, 0, 0, 2 /*eject+stop*/, 0,
142 0,0,0,0,0,0,0,0,0,0
143 };
144 rc = DRVHostBaseScsiCmd(pThis, abCmd, 6, PDMBLOCKTXDIR_NONE, NULL, NULL, NULL, 0, 0);
145
146#elif defined(RT_OS_LINUX)
147 rc = ioctl(pThis->FileDevice, CDROMEJECT, 0);
148 if (rc < 0)
149 {
150 if (errno == EBUSY)
151 rc = VERR_PDM_MEDIA_LOCKED;
152 else if (errno == ENOSYS)
153 rc = VERR_NOT_SUPPORTED;
154 else
155 rc = RTErrConvertFromErrno(errno);
156 }
157
158#elif defined(RT_OS_SOLARIS)
159 rc = ioctl(pThis->FileRawDevice, DKIOCEJECT, 0);
160 if (rc < 0)
161 {
162 if (errno == EBUSY)
163 rc = VERR_PDM_MEDIA_LOCKED;
164 else if (errno == ENOSYS || errno == ENOTSUP)
165 rc = VERR_NOT_SUPPORTED;
166 else if (errno == ENODEV)
167 rc = VERR_PDM_MEDIA_NOT_MOUNTED;
168 else
169 rc = RTErrConvertFromErrno(errno);
170 }
171
172#elif defined(RT_OS_WINDOWS)
173 RTFILE FileDevice = pThis->FileDevice;
174 if (FileDevice == NIL_RTFILE) /* obsolete crap */
175 rc = RTFileOpen(&FileDevice, pThis->pszDeviceOpen, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
176 if (RT_SUCCESS(rc))
177 {
178 /* do ioctl */
179 DWORD cbReturned;
180 if (DeviceIoControl((HANDLE)FileDevice, IOCTL_STORAGE_EJECT_MEDIA,
181 NULL, 0,
182 NULL, 0, &cbReturned,
183 NULL))
184 rc = VINF_SUCCESS;
185 else
186 rc = RTErrConvertFromWin32(GetLastError());
187
188 /* clean up handle */
189 if (FileDevice != pThis->FileDevice)
190 RTFileClose(FileDevice);
191 }
192 else
193 AssertMsgFailed(("Failed to open '%s' for ejecting this tray.\n", rc));
194
195
196#else
197 AssertMsgFailed(("Eject is not implemented!\n"));
198 rc = VINF_SUCCESS;
199#endif
200
201 /*
202 * Media is no longer present.
203 */
204 DRVHostBaseMediaNotPresent(pThis); /** @todo This isn't thread safe! */
205 }
206 else
207 {
208 Log(("drvHostDvdUnmount: Locked\n"));
209 rc = VERR_PDM_MEDIA_LOCKED;
210 }
211
212 RTCritSectLeave(&pThis->CritSect);
213 LogFlow(("drvHostDvdUnmount: returns %Rrc\n", rc));
214 return rc;
215}
216
217
218/**
219 * Locks or unlocks the drive.
220 *
221 * @returns VBox status code.
222 * @param pThis The instance data.
223 * @param fLock True if the request is to lock the drive, false if to unlock.
224 */
225static DECLCALLBACK(int) drvHostDvdDoLock(PDRVHOSTBASE pThis, bool fLock)
226{
227#ifdef RT_OS_DARWIN
228 uint8_t abCmd[16] =
229 {
230 SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL, 0, 0, 0, fLock, 0,
231 0,0,0,0,0,0,0,0,0,0
232 };
233 int rc = DRVHostBaseScsiCmd(pThis, abCmd, 6, PDMBLOCKTXDIR_NONE, NULL, NULL, NULL, 0, 0);
234
235#elif defined(RT_OS_LINUX)
236 int rc = ioctl(pThis->FileDevice, CDROM_LOCKDOOR, (int)fLock);
237 if (rc < 0)
238 {
239 if (errno == EBUSY)
240 rc = VERR_ACCESS_DENIED;
241 else if (errno == EDRIVE_CANT_DO_THIS)
242 rc = VERR_NOT_SUPPORTED;
243 else
244 rc = RTErrConvertFromErrno(errno);
245 }
246
247#elif defined(RT_OS_SOLARIS)
248 int rc = ioctl(pThis->FileRawDevice, fLock ? DKIOCLOCK : DKIOCUNLOCK, 0);
249 if (rc < 0)
250 {
251 if (errno == EBUSY)
252 rc = VERR_ACCESS_DENIED;
253 else if (errno == ENOTSUP || errno == ENOSYS)
254 rc = VERR_NOT_SUPPORTED;
255 else
256 rc = RTErrConvertFromErrno(errno);
257 }
258
259#elif defined(RT_OS_WINDOWS)
260
261 PREVENT_MEDIA_REMOVAL PreventMediaRemoval = {fLock};
262 DWORD cbReturned;
263 int rc;
264 if (DeviceIoControl((HANDLE)pThis->FileDevice, IOCTL_STORAGE_MEDIA_REMOVAL,
265 &PreventMediaRemoval, sizeof(PreventMediaRemoval),
266 NULL, 0, &cbReturned,
267 NULL))
268 rc = VINF_SUCCESS;
269 else
270 /** @todo figure out the return codes for already locked. */
271 rc = RTErrConvertFromWin32(GetLastError());
272
273#else
274 AssertMsgFailed(("Lock/Unlock is not implemented!\n"));
275 int rc = VINF_SUCCESS;
276
277#endif
278
279 LogFlow(("drvHostDvdDoLock(, fLock=%RTbool): returns %Rrc\n", fLock, rc));
280 return rc;
281}
282
283
284
285#ifdef RT_OS_LINUX
286/**
287 * Get the media size.
288 *
289 * @returns VBox status code.
290 * @param pThis The instance data.
291 * @param pcb Where to store the size.
292 */
293static int drvHostDvdGetMediaSize(PDRVHOSTBASE pThis, uint64_t *pcb)
294{
295 /*
296 * Query the media size.
297 */
298 /* Clear the media-changed-since-last-call-thingy just to be on the safe side. */
299 ioctl(pThis->FileDevice, CDROM_MEDIA_CHANGED, CDSL_CURRENT);
300 return RTFileSeek(pThis->FileDevice, 0, RTFILE_SEEK_END, pcb);
301
302}
303#endif /* RT_OS_LINUX */
304
305
306#ifdef USE_MEDIA_POLLING
307/**
308 * Do media change polling.
309 */
310DECLCALLBACK(int) drvHostDvdPoll(PDRVHOSTBASE pThis)
311{
312 /*
313 * Poll for media change.
314 */
315#if defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
316#ifdef RT_OS_DARWIN
317 AssertReturn(pThis->ppScsiTaskDI, VERR_INTERNAL_ERROR);
318#endif
319
320 /*
321 * Issue a TEST UNIT READY request.
322 */
323 bool fMediaChanged = false;
324 bool fMediaPresent = false;
325 uint8_t abCmd[16] = { SCSI_TEST_UNIT_READY, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 };
326 uint8_t abSense[32];
327 int rc2 = DRVHostBaseScsiCmd(pThis, abCmd, 6, PDMBLOCKTXDIR_NONE, NULL, NULL, abSense, sizeof(abSense), 0);
328 if (RT_SUCCESS(rc2))
329 fMediaPresent = true;
330 else if ( rc2 == VERR_UNRESOLVED_ERROR
331 && abSense[2] == 6 /* unit attention */
332 && ( (abSense[12] == 0x29 && abSense[13] < 5 /* reset */)
333 || (abSense[12] == 0x2a && abSense[13] == 0 /* parameters changed */) //???
334 || (abSense[12] == 0x3f && abSense[13] == 0 /* target operating conditions have changed */) //???
335 || (abSense[12] == 0x3f && abSense[13] == 2 /* changed operating definition */) //???
336 || (abSense[12] == 0x3f && abSense[13] == 3 /* inquery parameters changed */)
337 || (abSense[12] == 0x3f && abSense[13] == 5 /* device identifier changed */)
338 )
339 )
340 {
341 fMediaPresent = false;
342 fMediaChanged = true;
343 /** @todo check this media change stuff on Darwin. */
344 }
345
346#elif defined(RT_OS_LINUX)
347 bool fMediaPresent = ioctl(pThis->FileDevice, CDROM_DRIVE_STATUS, CDSL_CURRENT) == CDS_DISC_OK;
348
349#elif defined(RT_OS_SOLARIS)
350 bool fMediaPresent = false;
351 bool fMediaChanged = false;
352
353 /* Need to pass the previous state and DKIO_NONE for the first time. */
354 static dkio_state s_DeviceState = DKIO_NONE;
355 dkio_state PreviousState = s_DeviceState;
356 int rc2 = ioctl(pThis->FileRawDevice, DKIOCSTATE, &s_DeviceState);
357 if (rc2 == 0)
358 {
359 fMediaPresent = (s_DeviceState == DKIO_INSERTED);
360 if (PreviousState != s_DeviceState)
361 fMediaChanged = true;
362 }
363
364#else
365# error "Unsupported platform."
366#endif
367
368 RTCritSectEnter(&pThis->CritSect);
369
370 int rc = VINF_SUCCESS;
371 if (pThis->fMediaPresent != fMediaPresent)
372 {
373 LogFlow(("drvHostDvdPoll: %d -> %d\n", pThis->fMediaPresent, fMediaPresent));
374 pThis->fMediaPresent = false;
375 if (fMediaPresent)
376 rc = DRVHostBaseMediaPresent(pThis);
377 else
378 DRVHostBaseMediaNotPresent(pThis);
379 }
380 else if (fMediaPresent)
381 {
382 /*
383 * Poll for media change.
384 */
385#if defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
386 /* taken care of above. */
387#elif defined(RT_OS_LINUX)
388 bool fMediaChanged = ioctl(pThis->FileDevice, CDROM_MEDIA_CHANGED, CDSL_CURRENT) == 1;
389#else
390# error "Unsupported platform."
391#endif
392 if (fMediaChanged)
393 {
394 LogFlow(("drvHostDVDMediaThread: Media changed!\n"));
395 DRVHostBaseMediaNotPresent(pThis);
396 rc = DRVHostBaseMediaPresent(pThis);
397 }
398 }
399
400 RTCritSectLeave(&pThis->CritSect);
401 return rc;
402}
403#endif /* USE_MEDIA_POLLING */
404
405
406/** @copydoc PDMIBLOCK::pfnSendCmd */
407static int drvHostDvdSendCmd(PPDMIBLOCK pInterface, const uint8_t *pbCmd,
408 PDMBLOCKTXDIR enmTxDir, void *pvBuf, uint32_t *pcbBuf,
409 uint8_t *pabSense, size_t cbSense, uint32_t cTimeoutMillies)
410{
411 PDRVHOSTBASE pThis = PDMIBLOCK_2_DRVHOSTBASE(pInterface);
412 int rc;
413 LogFlow(("%s: cmd[0]=%#04x txdir=%d pcbBuf=%d timeout=%d\n", __FUNCTION__, pbCmd[0], enmTxDir, *pcbBuf, cTimeoutMillies));
414
415#if defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
416 /*
417 * Pass the request on to the internal scsi command interface.
418 * The command seems to be 12 bytes long, the docs a bit copy&pasty on the command length point...
419 */
420 if (enmTxDir == PDMBLOCKTXDIR_FROM_DEVICE)
421 memset(pvBuf, '\0', *pcbBuf); /* we got read size, but zero it anyway. */
422 rc = DRVHostBaseScsiCmd(pThis, pbCmd, 12, PDMBLOCKTXDIR_FROM_DEVICE, pvBuf, pcbBuf, pabSense, cbSense, cTimeoutMillies);
423 if (rc == VERR_UNRESOLVED_ERROR)
424 /* sense information set */
425 rc = VERR_DEV_IO_ERROR;
426
427#elif defined(RT_OS_L4)
428 /* Not really ported to L4 yet. */
429 rc = VERR_INTERNAL_ERROR;
430
431#elif defined(RT_OS_LINUX)
432 int direction;
433 struct cdrom_generic_command cgc;
434
435 switch (enmTxDir)
436 {
437 case PDMBLOCKTXDIR_NONE:
438 Assert(*pcbBuf == 0);
439 direction = CGC_DATA_NONE;
440 break;
441 case PDMBLOCKTXDIR_FROM_DEVICE:
442 Assert(*pcbBuf != 0);
443 Assert(*pcbBuf <= SCSI_MAX_BUFFER_SIZE);
444 /* Make sure that the buffer is clear for commands reading
445 * data. The actually received data may be shorter than what
446 * we expect, and due to the unreliable feedback about how much
447 * data the ioctl actually transferred, it's impossible to
448 * prevent that. Returning previous buffer contents may cause
449 * security problems inside the guest OS, if users can issue
450 * commands to the CDROM device. */
451 memset(pThis->pbDoubleBuffer, '\0', *pcbBuf);
452 direction = CGC_DATA_READ;
453 break;
454 case PDMBLOCKTXDIR_TO_DEVICE:
455 Assert(*pcbBuf != 0);
456 Assert(*pcbBuf <= SCSI_MAX_BUFFER_SIZE);
457 memcpy(pThis->pbDoubleBuffer, pvBuf, *pcbBuf);
458 direction = CGC_DATA_WRITE;
459 break;
460 default:
461 AssertMsgFailed(("enmTxDir invalid!\n"));
462 direction = CGC_DATA_NONE;
463 }
464 memset(&cgc, '\0', sizeof(cgc));
465 memcpy(cgc.cmd, pbCmd, CDROM_PACKET_SIZE);
466 cgc.buffer = (unsigned char *)pThis->pbDoubleBuffer;
467 cgc.buflen = *pcbBuf;
468 cgc.stat = 0;
469 Assert(cbSense >= sizeof(struct request_sense));
470 cgc.sense = (struct request_sense *)pabSense;
471 cgc.data_direction = direction;
472 cgc.quiet = false;
473 cgc.timeout = cTimeoutMillies;
474 rc = ioctl(pThis->FileDevice, CDROM_SEND_PACKET, &cgc);
475 if (rc < 0)
476 {
477 if (errno == EBUSY)
478 rc = VERR_PDM_MEDIA_LOCKED;
479 else if (errno == ENOSYS)
480 rc = VERR_NOT_SUPPORTED;
481 else
482 {
483 rc = RTErrConvertFromErrno(errno);
484 if (rc == VERR_ACCESS_DENIED && cgc.sense->sense_key == SCSI_SENSE_NONE)
485 cgc.sense->sense_key = SCSI_SENSE_ILLEGAL_REQUEST;
486 Log2(("%s: error status %d, rc=%Rrc\n", __FUNCTION__, cgc.stat, rc));
487 }
488 }
489 switch (enmTxDir)
490 {
491 case PDMBLOCKTXDIR_FROM_DEVICE:
492 memcpy(pvBuf, pThis->pbDoubleBuffer, *pcbBuf);
493 break;
494 default:
495 ;
496 }
497 Log2(("%s: after ioctl: cgc.buflen=%d txlen=%d\n", __FUNCTION__, cgc.buflen, *pcbBuf));
498 /* The value of cgc.buflen does not reliably reflect the actual amount
499 * of data transferred (for packet commands with little data transfer
500 * it's 0). So just assume that everything worked ok. */
501
502#elif defined(RT_OS_SOLARIS)
503 struct uscsi_cmd usc;
504 union scsi_cdb scdb;
505 memset(&usc, 0, sizeof(struct uscsi_cmd));
506 memset(&scdb, 0, sizeof(scdb));
507
508 switch (enmTxDir)
509 {
510 case PDMBLOCKTXDIR_NONE:
511 Assert(*pcbBuf == 0);
512 usc.uscsi_flags = USCSI_READ;
513 /* nothing to do */
514 break;
515
516 case PDMBLOCKTXDIR_FROM_DEVICE:
517 Assert(*pcbBuf != 0);
518 /* Make sure that the buffer is clear for commands reading
519 * data. The actually received data may be shorter than what
520 * we expect, and due to the unreliable feedback about how much
521 * data the ioctl actually transferred, it's impossible to
522 * prevent that. Returning previous buffer contents may cause
523 * security problems inside the guest OS, if users can issue
524 * commands to the CDROM device. */
525 memset(pvBuf, '\0', *pcbBuf);
526 usc.uscsi_flags = USCSI_READ;
527 break;
528 case PDMBLOCKTXDIR_TO_DEVICE:
529 Assert(*pcbBuf != 0);
530 usc.uscsi_flags = USCSI_WRITE;
531 break;
532 default:
533 AssertMsgFailedReturn(("%d\n", enmTxDir), VERR_INTERNAL_ERROR);
534 }
535 usc.uscsi_flags |= USCSI_RQENABLE;
536 usc.uscsi_rqbuf = (char *)pabSense;
537 usc.uscsi_rqlen = cbSense;
538 usc.uscsi_cdb = (caddr_t)&scdb;
539 usc.uscsi_cdblen = 12;
540 memcpy (usc.uscsi_cdb, pbCmd, usc.uscsi_cdblen);
541 usc.uscsi_bufaddr = (caddr_t)pvBuf;
542 usc.uscsi_buflen = *pcbBuf;
543 usc.uscsi_timeout = (cTimeoutMillies + 999) / 1000;
544
545 /* We need root privileges for user-SCSI under Solaris. */
546#ifdef VBOX_WITH_SUID_WRAPPER
547 uid_t effUserID = geteuid();
548 solarisEnterRootMode(&effUserID); /** @todo check return code when this really works. */
549#endif
550 rc = ioctl(pThis->FileRawDevice, USCSICMD, &usc);
551#ifdef VBOX_WITH_SUID_WRAPPER
552 solarisExitRootMode(&effUserID);
553#endif
554 if (rc < 0)
555 {
556 if (errno == EPERM)
557 return VERR_PERMISSION_DENIED;
558 if (usc.uscsi_status)
559 {
560 rc = RTErrConvertFromErrno(errno);
561 Log2(("%s: error status. rc=%Rrc\n", __FUNCTION__, rc));
562 }
563 }
564 Log2(("%s: after ioctl: residual buflen=%d original buflen=%d\n", __FUNCTION__, usc.uscsi_resid, usc.uscsi_buflen));
565
566#elif defined(RT_OS_WINDOWS)
567 int direction;
568 struct _REQ
569 {
570 SCSI_PASS_THROUGH_DIRECT spt;
571 uint8_t aSense[64];
572 } Req;
573 DWORD cbReturned = 0;
574
575 switch (enmTxDir)
576 {
577 case PDMBLOCKTXDIR_NONE:
578 direction = SCSI_IOCTL_DATA_UNSPECIFIED;
579 break;
580 case PDMBLOCKTXDIR_FROM_DEVICE:
581 Assert(*pcbBuf != 0);
582 /* Make sure that the buffer is clear for commands reading
583 * data. The actually received data may be shorter than what
584 * we expect, and due to the unreliable feedback about how much
585 * data the ioctl actually transferred, it's impossible to
586 * prevent that. Returning previous buffer contents may cause
587 * security problems inside the guest OS, if users can issue
588 * commands to the CDROM device. */
589 memset(pvBuf, '\0', *pcbBuf);
590 direction = SCSI_IOCTL_DATA_IN;
591 break;
592 case PDMBLOCKTXDIR_TO_DEVICE:
593 direction = SCSI_IOCTL_DATA_OUT;
594 break;
595 default:
596 AssertMsgFailed(("enmTxDir invalid!\n"));
597 direction = SCSI_IOCTL_DATA_UNSPECIFIED;
598 }
599 memset(&Req, '\0', sizeof(Req));
600 Req.spt.Length = sizeof(Req.spt);
601 Req.spt.CdbLength = 12;
602 memcpy(Req.spt.Cdb, pbCmd, Req.spt.CdbLength);
603 Req.spt.DataBuffer = pvBuf;
604 Req.spt.DataTransferLength = *pcbBuf;
605 Req.spt.DataIn = direction;
606 Req.spt.TimeOutValue = (cTimeoutMillies + 999) / 1000; /* Convert to seconds */
607 Assert(cbSense <= sizeof(Req.aSense));
608 Req.spt.SenseInfoLength = (UCHAR)RT_MIN(sizeof(Req.aSense), cbSense);
609 Req.spt.SenseInfoOffset = RT_OFFSETOF(struct _REQ, aSense);
610 if (DeviceIoControl((HANDLE)pThis->FileDevice, IOCTL_SCSI_PASS_THROUGH_DIRECT,
611 &Req, sizeof(Req), &Req, sizeof(Req), &cbReturned, NULL))
612 {
613 if (cbReturned > RT_OFFSETOF(struct _REQ, aSense))
614 memcpy(pabSense, Req.aSense, cbSense);
615 else
616 memset(pabSense, '\0', cbSense);
617 /* Windows shares the property of not properly reflecting the actually
618 * transferred data size. See above. Assume that everything worked ok.
619 * Except if there are sense information. */
620 rc = (pabSense[2] & 0x0f) == SCSI_SENSE_NONE
621 ? VINF_SUCCESS
622 : VERR_DEV_IO_ERROR;
623 }
624 else
625 rc = RTErrConvertFromWin32(GetLastError());
626 Log2(("%s: scsistatus=%d bytes returned=%d tlength=%d\n", __FUNCTION__, Req.spt.ScsiStatus, cbReturned, Req.spt.DataTransferLength));
627
628#else
629# error "Unsupported platform."
630#endif
631
632 if (pbCmd[0] == SCSI_GET_EVENT_STATUS_NOTIFICATION)
633 {
634 uint8_t *pbBuf = (uint8_t*)pvBuf;
635 Log2(("Event Status Notification class=%#02x supported classes=%#02x\n", pbBuf[2], pbBuf[3]));
636 if (RT_BE2H_U16(*(uint16_t*)pbBuf) >= 6)
637 Log2((" event %#02x %#02x %#02x %#02x\n", pbBuf[4], pbBuf[5], pbBuf[6], pbBuf[7]));
638 }
639
640 LogFlow(("%s: rc=%Rrc\n", __FUNCTION__, rc));
641 return rc;
642}
643
644
645#ifdef VBOX_WITH_SUID_WRAPPER
646/* These functions would have to go into a seperate solaris binary with
647 * the setuid permission set, which would run the user-SCSI ioctl and
648 * return the value. BUT... this might be prohibitively slow.
649 */
650# ifdef RT_OS_SOLARIS
651
652/**
653 * Checks if the current user is authorized using Solaris' role-based access control.
654 * Made as a seperate function with so that it need not be invoked each time we need
655 * to gain root access.
656 *
657 * @returns VBox error code.
658 */
659static int solarisCheckUserAuth()
660{
661 /* Uses Solaris' role-based access control (RBAC).*/
662 struct passwd *pPass = getpwuid(getuid());
663 if (pPass == NULL || chkauthattr("solaris.device.cdrw", pPass->pw_name) == 0)
664 return VERR_PERMISSION_DENIED;
665
666 return VINF_SUCCESS;
667}
668
669
670/**
671 * Setuid wrapper to gain root access.
672 *
673 * @returns VBox error code.
674 * @param pEffUserID Pointer to effective user ID.
675 */
676static int solarisEnterRootMode(uid_t *pEffUserID)
677{
678 /* Increase privilege if required */
679 if (*pEffUserID != 0)
680 {
681 if (seteuid(0) == 0)
682 {
683 *pEffUserID = 0;
684 return VINF_SUCCESS;
685 }
686 return VERR_PERMISSION_DENIED;
687 }
688 return VINF_SUCCESS;
689}
690
691
692/**
693 * Setuid wrapper to relinquish root access.
694 *
695 * @returns VBox error code.
696 * @param pEffUserID Pointer to effective user ID.
697 */
698static int solarisExitRootMode(uid_t *pEffUserID)
699{
700 /* Get back to user mode. */
701 if (*pEffUserID == 0)
702 {
703 uid_t realID = getuid();
704 if (seteuid(realID) == 0)
705 {
706 *pEffUserID = realID;
707 return VINF_SUCCESS;
708 }
709 return VERR_PERMISSION_DENIED;
710 }
711 return VINF_SUCCESS;
712}
713
714# endif /* RT_OS_SOLARIS */
715#endif /* VBOX_WITH_SUID_WRAPPER */
716
717
718/* -=-=-=-=- driver interface -=-=-=-=- */
719
720
721/** @copydoc FNPDMDRVDESTRUCT */
722DECLCALLBACK(void) drvHostDvdDestruct(PPDMDRVINS pDrvIns)
723{
724#ifdef RT_OS_LINUX
725 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
726
727 if (pThis->pbDoubleBuffer)
728 {
729 RTMemFree(pThis->pbDoubleBuffer);
730 pThis->pbDoubleBuffer = NULL;
731 }
732#endif
733 return DRVHostBaseDestruct(pDrvIns);
734}
735
736
737/**
738 * Construct a host dvd drive driver instance.
739 *
740 * @copydoc FNPDMDRVCONSTRUCT
741 */
742static DECLCALLBACK(int) drvHostDvdConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
743{
744 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
745 LogFlow(("drvHostDvdConstruct: iInstance=%d\n", pDrvIns->iInstance));
746
747 /*
748 * Validate configuration.
749 */
750 if (!CFGMR3AreValuesValid(pCfg, "Path\0Interval\0Locked\0BIOSVisible\0AttachFailError\0Passthrough\0"))
751 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
752
753
754 /*
755 * Init instance data.
756 */
757 int rc = DRVHostBaseInitData(pDrvIns, pCfg, PDMBLOCKTYPE_DVD);
758 if (RT_SUCCESS(rc))
759 {
760 /*
761 * Override stuff.
762 */
763#ifdef RT_OS_LINUX
764 pThis->pbDoubleBuffer = (uint8_t *)RTMemAlloc(SCSI_MAX_BUFFER_SIZE);
765 if (!pThis->pbDoubleBuffer)
766 return VERR_NO_MEMORY;
767#endif
768
769#ifndef RT_OS_L4 /* Passthrough is not supported on L4 yet */
770 bool fPassthrough;
771 rc = CFGMR3QueryBool(pCfg, "Passthrough", &fPassthrough);
772 if (RT_SUCCESS(rc) && fPassthrough)
773 {
774 pThis->IBlock.pfnSendCmd = drvHostDvdSendCmd;
775 /* Passthrough requires opening the device in R/W mode. */
776 pThis->fReadOnlyConfig = false;
777# ifdef VBOX_WITH_SUID_WRAPPER /* Solaris setuid for Passthrough mode. */
778 rc = solarisCheckUserAuth();
779 if (RT_FAILURE(rc))
780 {
781 Log(("DVD: solarisCheckUserAuth failed. Permission denied!\n"));
782 return rc;
783 }
784# endif /* VBOX_WITH_SUID_WRAPPER */
785 }
786#endif /* !RT_OS_L4 */
787
788 pThis->IMount.pfnUnmount = drvHostDvdUnmount;
789 pThis->pfnDoLock = drvHostDvdDoLock;
790#ifdef USE_MEDIA_POLLING
791 if (!fPassthrough)
792 pThis->pfnPoll = drvHostDvdPoll;
793 else
794 pThis->pfnPoll = NULL;
795#endif
796#ifdef RT_OS_LINUX
797 pThis->pfnGetMediaSize = drvHostDvdGetMediaSize;
798#endif
799
800 /*
801 * 2nd init part.
802 */
803 rc = DRVHostBaseInitFinish(pThis);
804 }
805 if (RT_FAILURE(rc))
806 {
807 if (!pThis->fAttachFailError)
808 {
809 /* Suppressing the attach failure error must not affect the normal
810 * DRVHostBaseDestruct, so reset this flag below before leaving. */
811 pThis->fKeepInstance = true;
812 rc = VINF_SUCCESS;
813 }
814 DRVHostBaseDestruct(pDrvIns);
815 pThis->fKeepInstance = false;
816 }
817
818 LogFlow(("drvHostDvdConstruct: returns %Rrc\n", rc));
819 return rc;
820}
821
822
823/**
824 * Block driver registration record.
825 */
826const PDMDRVREG g_DrvHostDVD =
827{
828 /* u32Version */
829 PDM_DRVREG_VERSION,
830 /* szName */
831 "HostDVD",
832 /* szRCMod */
833 "",
834 /* szR0Mod */
835 "",
836 /* pszDescription */
837 "Host DVD Block Driver.",
838 /* fFlags */
839 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
840 /* fClass. */
841 PDM_DRVREG_CLASS_BLOCK,
842 /* cMaxInstances */
843 ~0,
844 /* cbInstance */
845 sizeof(DRVHOSTBASE),
846 /* pfnConstruct */
847 drvHostDvdConstruct,
848 /* pfnDestruct */
849 drvHostDvdDestruct,
850 /* pfnRelocate */
851 NULL,
852 /* pfnIOCtl */
853 NULL,
854 /* pfnPowerOn */
855 NULL,
856 /* pfnReset */
857 NULL,
858 /* pfnSuspend */
859 NULL,
860 /* pfnResume */
861 NULL,
862 /* pfnAttach */
863 NULL,
864 /* pfnDetach */
865 NULL,
866 /* pfnPowerOff */
867 NULL,
868 /* pfnSoftReset */
869 NULL,
870 /* u32EndVersion */
871 PDM_DRVREG_VERSION
872};
873
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