VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvHostBase.cpp@ 51578

Last change on this file since 51578 was 51578, checked in by vboxsync, 11 years ago

Mac OS X host: improve support for host optical drives; add support for CD and BlueRay drives.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 76.2 KB
Line 
1/* $Id: DrvHostBase.cpp 51578 2014-06-09 09:45:16Z vboxsync $ */
2/** @file
3 * DrvHostBase - Host base drive access driver.
4 */
5
6/*
7 * Copyright (C) 2006-2014 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#ifdef RT_OS_DARWIN
24# include <mach/mach.h>
25# include <Carbon/Carbon.h>
26# include <IOKit/IOKitLib.h>
27# include <IOKit/storage/IOStorageDeviceCharacteristics.h>
28# include <IOKit/scsi/SCSITaskLib.h>
29# include <IOKit/scsi/SCSICommandOperationCodes.h>
30# include <IOKit/IOBSD.h>
31# include <DiskArbitration/DiskArbitration.h>
32# include <mach/mach_error.h>
33# include <VBox/scsi.h>
34
35#elif defined(RT_OS_L4)
36 /* Nothing special requires... yeah, right. */
37
38#elif defined(RT_OS_LINUX)
39# include <sys/ioctl.h>
40# include <sys/fcntl.h>
41# include <errno.h>
42
43#elif defined(RT_OS_SOLARIS)
44# include <fcntl.h>
45# include <errno.h>
46# include <stropts.h>
47# include <malloc.h>
48# include <sys/dkio.h>
49extern "C" char *getfullblkname(char *);
50
51#elif defined(RT_OS_WINDOWS)
52# define WIN32_NO_STATUS
53# include <Windows.h>
54# include <dbt.h>
55# undef WIN32_NO_STATUS
56# include <ntstatus.h>
57
58/* from ntdef.h */
59typedef LONG NTSTATUS;
60
61/* from ntddk.h */
62typedef struct _IO_STATUS_BLOCK {
63 union {
64 NTSTATUS Status;
65 PVOID Pointer;
66 };
67 ULONG_PTR Information;
68} IO_STATUS_BLOCK, *PIO_STATUS_BLOCK;
69
70
71/* from ntinternals.com */
72typedef enum _FS_INFORMATION_CLASS {
73 FileFsVolumeInformation=1,
74 FileFsLabelInformation,
75 FileFsSizeInformation,
76 FileFsDeviceInformation,
77 FileFsAttributeInformation,
78 FileFsControlInformation,
79 FileFsFullSizeInformation,
80 FileFsObjectIdInformation,
81 FileFsMaximumInformation
82} FS_INFORMATION_CLASS, *PFS_INFORMATION_CLASS;
83
84typedef struct _FILE_FS_SIZE_INFORMATION {
85 LARGE_INTEGER TotalAllocationUnits;
86 LARGE_INTEGER AvailableAllocationUnits;
87 ULONG SectorsPerAllocationUnit;
88 ULONG BytesPerSector;
89} FILE_FS_SIZE_INFORMATION, *PFILE_FS_SIZE_INFORMATION;
90
91extern "C"
92NTSTATUS __stdcall NtQueryVolumeInformationFile(
93 /*IN*/ HANDLE FileHandle,
94 /*OUT*/ PIO_STATUS_BLOCK IoStatusBlock,
95 /*OUT*/ PVOID FileSystemInformation,
96 /*IN*/ ULONG Length,
97 /*IN*/ FS_INFORMATION_CLASS FileSystemInformationClass );
98
99#elif defined(RT_OS_FREEBSD)
100# include <sys/cdefs.h>
101# include <sys/param.h>
102# include <errno.h>
103# include <stdio.h>
104# include <cam/cam.h>
105# include <cam/cam_ccb.h>
106# include <cam/scsi/scsi_message.h>
107# include <cam/scsi/scsi_pass.h>
108# include <VBox/scsi.h>
109# include <iprt/log.h>
110#else
111# error "Unsupported Platform."
112#endif
113
114#include <VBox/vmm/pdmdrv.h>
115#include <iprt/assert.h>
116#include <iprt/file.h>
117#include <iprt/path.h>
118#include <iprt/string.h>
119#include <iprt/thread.h>
120#include <iprt/semaphore.h>
121#include <iprt/uuid.h>
122#include <iprt/asm.h>
123#include <iprt/critsect.h>
124#include <iprt/ctype.h>
125
126#include "DrvHostBase.h"
127
128
129
130
131/* -=-=-=-=- IBlock -=-=-=-=- */
132
133/** @copydoc PDMIBLOCK::pfnRead */
134static DECLCALLBACK(int) drvHostBaseRead(PPDMIBLOCK pInterface, uint64_t off, void *pvBuf, size_t cbRead)
135{
136 PDRVHOSTBASE pThis = PDMIBLOCK_2_DRVHOSTBASE(pInterface);
137 LogFlow(("%s-%d: drvHostBaseRead: off=%#llx pvBuf=%p cbRead=%#x (%s)\n",
138 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, pvBuf, cbRead, pThis->pszDevice));
139 RTCritSectEnter(&pThis->CritSect);
140
141 /*
142 * Check the state.
143 */
144 int rc;
145#ifdef RT_OS_DARWIN
146 if ( pThis->fMediaPresent
147 && pThis->ppScsiTaskDI
148 && pThis->cbBlock)
149#elif RT_OS_FREEBSD
150 if ( pThis->fMediaPresent
151 && pThis->cbBlock)
152#else
153 if (pThis->fMediaPresent)
154#endif
155 {
156#if defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
157 /*
158 * Issue a READ(12) request.
159 */
160 do
161 {
162 const uint32_t LBA = off / pThis->cbBlock;
163 AssertReturn(!(off % pThis->cbBlock), VERR_INVALID_PARAMETER);
164 uint32_t cbRead32 = cbRead > SCSI_MAX_BUFFER_SIZE
165 ? SCSI_MAX_BUFFER_SIZE
166 : (uint32_t)cbRead;
167 const uint32_t cBlocks = cbRead32 / pThis->cbBlock;
168 AssertReturn(!(cbRead % pThis->cbBlock), VERR_INVALID_PARAMETER);
169 uint8_t abCmd[16] =
170 {
171 SCSI_READ_12, 0,
172 RT_BYTE4(LBA), RT_BYTE3(LBA), RT_BYTE2(LBA), RT_BYTE1(LBA),
173 RT_BYTE4(cBlocks), RT_BYTE3(cBlocks), RT_BYTE2(cBlocks), RT_BYTE1(cBlocks),
174 0, 0, 0, 0, 0
175 };
176 rc = DRVHostBaseScsiCmd(pThis, abCmd, 12, PDMBLOCKTXDIR_FROM_DEVICE, pvBuf, &cbRead32, NULL, 0, 0);
177
178 off += cbRead32;
179 cbRead -= cbRead32;
180 pvBuf = (uint8_t *)pvBuf + cbRead32;
181 } while ((cbRead > 0) && RT_SUCCESS(rc));
182
183#else
184 /*
185 * Seek and read.
186 */
187 rc = RTFileReadAt(pThis->hFileDevice, off, pvBuf, cbRead, NULL);
188 if (RT_SUCCESS(rc))
189 {
190 Log2(("%s-%d: drvHostBaseRead: off=%#llx cbRead=%#x\n"
191 "%16.*Rhxd\n",
192 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, cbRead, cbRead, pvBuf));
193 }
194 else
195 Log(("%s-%d: drvHostBaseRead: RTFileReadAt(%RTfile, %#llx, %p, %#x) -> %Rrc ('%s')\n",
196 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->hFileDevice,
197 off, pvBuf, cbRead, rc, pThis->pszDevice));
198#endif
199 }
200 else
201 rc = VERR_MEDIA_NOT_PRESENT;
202
203 RTCritSectLeave(&pThis->CritSect);
204 LogFlow(("%s-%d: drvHostBaseRead: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
205 return rc;
206}
207
208
209/** @copydoc PDMIBLOCK::pfnWrite */
210static DECLCALLBACK(int) drvHostBaseWrite(PPDMIBLOCK pInterface, uint64_t off, const void *pvBuf, size_t cbWrite)
211{
212 PDRVHOSTBASE pThis = PDMIBLOCK_2_DRVHOSTBASE(pInterface);
213 LogFlow(("%s-%d: drvHostBaseWrite: off=%#llx pvBuf=%p cbWrite=%#x (%s)\n",
214 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, pvBuf, cbWrite, pThis->pszDevice));
215 Log2(("%s-%d: drvHostBaseWrite: off=%#llx cbWrite=%#x\n"
216 "%16.*Rhxd\n",
217 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, cbWrite, cbWrite, pvBuf));
218 RTCritSectEnter(&pThis->CritSect);
219
220 /*
221 * Check the state.
222 */
223 int rc;
224 if (!pThis->fReadOnly)
225 {
226 if (pThis->fMediaPresent)
227 {
228#if defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
229 /** @todo write support... */
230 rc = VERR_WRITE_PROTECT;
231
232#else
233 /*
234 * Seek and write.
235 */
236 rc = RTFileWriteAt(pThis->hFileDevice, off, pvBuf, cbWrite, NULL);
237 if (RT_FAILURE(rc))
238 Log(("%s-%d: drvHostBaseWrite: RTFileWriteAt(%RTfile, %#llx, %p, %#x) -> %Rrc ('%s')\n",
239 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->hFileDevice,
240 off, pvBuf, cbWrite, rc, pThis->pszDevice));
241#endif
242 }
243 else
244 rc = VERR_MEDIA_NOT_PRESENT;
245 }
246 else
247 rc = VERR_WRITE_PROTECT;
248
249 RTCritSectLeave(&pThis->CritSect);
250 LogFlow(("%s-%d: drvHostBaseWrite: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
251 return rc;
252}
253
254
255/** @copydoc PDMIBLOCK::pfnFlush */
256static DECLCALLBACK(int) drvHostBaseFlush(PPDMIBLOCK pInterface)
257{
258 int rc;
259 PDRVHOSTBASE pThis = PDMIBLOCK_2_DRVHOSTBASE(pInterface);
260 LogFlow(("%s-%d: drvHostBaseFlush: (%s)\n",
261 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDevice));
262 RTCritSectEnter(&pThis->CritSect);
263
264 if (pThis->fMediaPresent)
265 {
266#if defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
267 rc = VINF_SUCCESS;
268 /** @todo scsi device buffer flush... */
269#else
270 rc = RTFileFlush(pThis->hFileDevice);
271#endif
272 }
273 else
274 rc = VERR_MEDIA_NOT_PRESENT;
275
276 RTCritSectLeave(&pThis->CritSect);
277 LogFlow(("%s-%d: drvHostBaseFlush: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
278 return rc;
279}
280
281
282/** @copydoc PDMIBLOCK::pfnIsReadOnly */
283static DECLCALLBACK(bool) drvHostBaseIsReadOnly(PPDMIBLOCK pInterface)
284{
285 PDRVHOSTBASE pThis = PDMIBLOCK_2_DRVHOSTBASE(pInterface);
286 return pThis->fReadOnly;
287}
288
289
290/** @copydoc PDMIBLOCK::pfnGetSize */
291static DECLCALLBACK(uint64_t) drvHostBaseGetSize(PPDMIBLOCK pInterface)
292{
293 PDRVHOSTBASE pThis = PDMIBLOCK_2_DRVHOSTBASE(pInterface);
294 RTCritSectEnter(&pThis->CritSect);
295
296 uint64_t cb = 0;
297 if (pThis->fMediaPresent)
298 cb = pThis->cbSize;
299
300 RTCritSectLeave(&pThis->CritSect);
301 LogFlow(("%s-%d: drvHostBaseGetSize: returns %llu\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, cb));
302 return cb;
303}
304
305
306/** @copydoc PDMIBLOCK::pfnGetType */
307static DECLCALLBACK(PDMBLOCKTYPE) drvHostBaseGetType(PPDMIBLOCK pInterface)
308{
309 PDRVHOSTBASE pThis = PDMIBLOCK_2_DRVHOSTBASE(pInterface);
310 LogFlow(("%s-%d: drvHostBaseGetType: returns %d\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->enmType));
311 return pThis->enmType;
312}
313
314
315/** @copydoc PDMIBLOCK::pfnGetUuid */
316static DECLCALLBACK(int) drvHostBaseGetUuid(PPDMIBLOCK pInterface, PRTUUID pUuid)
317{
318 PDRVHOSTBASE pThis = PDMIBLOCK_2_DRVHOSTBASE(pInterface);
319
320 *pUuid = pThis->Uuid;
321
322 LogFlow(("%s-%d: drvHostBaseGetUuid: returns VINF_SUCCESS *pUuid=%RTuuid\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pUuid));
323 return VINF_SUCCESS;
324}
325
326
327/* -=-=-=-=- IBlockBios -=-=-=-=- */
328
329/** Makes a PDRVHOSTBASE out of a PPDMIBLOCKBIOS. */
330#define PDMIBLOCKBIOS_2_DRVHOSTBASE(pInterface) ( (PDRVHOSTBASE((uintptr_t)pInterface - RT_OFFSETOF(DRVHOSTBASE, IBlockBios))) )
331
332
333/** @copydoc PDMIBLOCKBIOS::pfnGetPCHSGeometry */
334static DECLCALLBACK(int) drvHostBaseGetPCHSGeometry(PPDMIBLOCKBIOS pInterface, PPDMMEDIAGEOMETRY pPCHSGeometry)
335{
336 PDRVHOSTBASE pThis = PDMIBLOCKBIOS_2_DRVHOSTBASE(pInterface);
337 RTCritSectEnter(&pThis->CritSect);
338
339 int rc = VINF_SUCCESS;
340 if (pThis->fMediaPresent)
341 {
342 if ( pThis->PCHSGeometry.cCylinders > 0
343 && pThis->PCHSGeometry.cHeads > 0
344 && pThis->PCHSGeometry.cSectors > 0)
345 {
346 *pPCHSGeometry = pThis->PCHSGeometry;
347 }
348 else
349 rc = VERR_PDM_GEOMETRY_NOT_SET;
350 }
351 else
352 rc = VERR_PDM_MEDIA_NOT_MOUNTED;
353
354 RTCritSectLeave(&pThis->CritSect);
355 LogFlow(("%s-%d: %s: returns %Rrc CHS={%d,%d,%d}\n",
356 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, __FUNCTION__, rc, pThis->PCHSGeometry.cCylinders, pThis->PCHSGeometry.cHeads, pThis->PCHSGeometry.cSectors));
357 return rc;
358}
359
360
361/** @copydoc PDMIBLOCKBIOS::pfnSetPCHSGeometry */
362static DECLCALLBACK(int) drvHostBaseSetPCHSGeometry(PPDMIBLOCKBIOS pInterface, PCPDMMEDIAGEOMETRY pPCHSGeometry)
363{
364 PDRVHOSTBASE pThis = PDMIBLOCKBIOS_2_DRVHOSTBASE(pInterface);
365 LogFlow(("%s-%d: %s: cCylinders=%d cHeads=%d cSectors=%d\n",
366 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, __FUNCTION__, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
367 RTCritSectEnter(&pThis->CritSect);
368
369 int rc = VINF_SUCCESS;
370 if (pThis->fMediaPresent)
371 {
372 pThis->PCHSGeometry = *pPCHSGeometry;
373 }
374 else
375 {
376 AssertMsgFailed(("Invalid state! Not mounted!\n"));
377 rc = VERR_PDM_MEDIA_NOT_MOUNTED;
378 }
379
380 RTCritSectLeave(&pThis->CritSect);
381 return rc;
382}
383
384
385/** @copydoc PDMIBLOCKBIOS::pfnGetLCHSGeometry */
386static DECLCALLBACK(int) drvHostBaseGetLCHSGeometry(PPDMIBLOCKBIOS pInterface, PPDMMEDIAGEOMETRY pLCHSGeometry)
387{
388 PDRVHOSTBASE pThis = PDMIBLOCKBIOS_2_DRVHOSTBASE(pInterface);
389 RTCritSectEnter(&pThis->CritSect);
390
391 int rc = VINF_SUCCESS;
392 if (pThis->fMediaPresent)
393 {
394 if ( pThis->LCHSGeometry.cCylinders > 0
395 && pThis->LCHSGeometry.cHeads > 0
396 && pThis->LCHSGeometry.cSectors > 0)
397 {
398 *pLCHSGeometry = pThis->LCHSGeometry;
399 }
400 else
401 rc = VERR_PDM_GEOMETRY_NOT_SET;
402 }
403 else
404 rc = VERR_PDM_MEDIA_NOT_MOUNTED;
405
406 RTCritSectLeave(&pThis->CritSect);
407 LogFlow(("%s-%d: %s: returns %Rrc CHS={%d,%d,%d}\n",
408 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, __FUNCTION__, rc, pThis->LCHSGeometry.cCylinders, pThis->LCHSGeometry.cHeads, pThis->LCHSGeometry.cSectors));
409 return rc;
410}
411
412
413/** @copydoc PDMIBLOCKBIOS::pfnSetLCHSGeometry */
414static DECLCALLBACK(int) drvHostBaseSetLCHSGeometry(PPDMIBLOCKBIOS pInterface, PCPDMMEDIAGEOMETRY pLCHSGeometry)
415{
416 PDRVHOSTBASE pThis = PDMIBLOCKBIOS_2_DRVHOSTBASE(pInterface);
417 LogFlow(("%s-%d: %s: cCylinders=%d cHeads=%d cSectors=%d\n",
418 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, __FUNCTION__, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
419 RTCritSectEnter(&pThis->CritSect);
420
421 int rc = VINF_SUCCESS;
422 if (pThis->fMediaPresent)
423 {
424 pThis->LCHSGeometry = *pLCHSGeometry;
425 }
426 else
427 {
428 AssertMsgFailed(("Invalid state! Not mounted!\n"));
429 rc = VERR_PDM_MEDIA_NOT_MOUNTED;
430 }
431
432 RTCritSectLeave(&pThis->CritSect);
433 return rc;
434}
435
436
437/** @copydoc PDMIBLOCKBIOS::pfnIsVisible */
438static DECLCALLBACK(bool) drvHostBaseIsVisible(PPDMIBLOCKBIOS pInterface)
439{
440 PDRVHOSTBASE pThis = PDMIBLOCKBIOS_2_DRVHOSTBASE(pInterface);
441 return pThis->fBiosVisible;
442}
443
444
445/** @copydoc PDMIBLOCKBIOS::pfnGetType */
446static DECLCALLBACK(PDMBLOCKTYPE) drvHostBaseBiosGetType(PPDMIBLOCKBIOS pInterface)
447{
448 PDRVHOSTBASE pThis = PDMIBLOCKBIOS_2_DRVHOSTBASE(pInterface);
449 return pThis->enmType;
450}
451
452
453
454/* -=-=-=-=- IMount -=-=-=-=- */
455
456/** @copydoc PDMIMOUNT::pfnMount */
457static DECLCALLBACK(int) drvHostBaseMount(PPDMIMOUNT pInterface, const char *pszFilename, const char *pszCoreDriver)
458{
459 /* We're not mountable. */
460 AssertMsgFailed(("drvHostBaseMount: This shouldn't be called!\n"));
461 return VERR_PDM_MEDIA_MOUNTED;
462}
463
464
465/** @copydoc PDMIMOUNT::pfnUnmount */
466static DECLCALLBACK(int) drvHostBaseUnmount(PPDMIMOUNT pInterface, bool fForce, bool fEject)
467{
468 /* While we're not mountable (see drvHostBaseMount), we're unmountable. */
469 PDRVHOSTBASE pThis = PDMIMOUNT_2_DRVHOSTBASE(pInterface);
470 RTCritSectEnter(&pThis->CritSect);
471
472 /*
473 * Validate state.
474 */
475 int rc = VINF_SUCCESS;
476 if (!pThis->fLocked || fForce)
477 {
478 /* Unlock drive if necessary. */
479 if (pThis->fLocked)
480 {
481 if (pThis->pfnDoLock)
482 rc = pThis->pfnDoLock(pThis, false);
483 if (RT_SUCCESS(rc))
484 pThis->fLocked = false;
485 }
486
487 /*
488 * Media is no longer present.
489 */
490 DRVHostBaseMediaNotPresent(pThis);
491 }
492 else
493 {
494 Log(("drvHostiBaseUnmount: Locked\n"));
495 rc = VERR_PDM_MEDIA_LOCKED;
496 }
497
498 RTCritSectLeave(&pThis->CritSect);
499 LogFlow(("drvHostBaseUnmount: returns %Rrc\n", rc));
500 return rc;
501}
502
503
504/** @copydoc PDMIMOUNT::pfnIsMounted */
505static DECLCALLBACK(bool) drvHostBaseIsMounted(PPDMIMOUNT pInterface)
506{
507 PDRVHOSTBASE pThis = PDMIMOUNT_2_DRVHOSTBASE(pInterface);
508 RTCritSectEnter(&pThis->CritSect);
509
510 bool fRc = pThis->fMediaPresent;
511
512 RTCritSectLeave(&pThis->CritSect);
513 return fRc;
514}
515
516
517/** @copydoc PDMIMOUNT::pfnIsLocked */
518static DECLCALLBACK(int) drvHostBaseLock(PPDMIMOUNT pInterface)
519{
520 PDRVHOSTBASE pThis = PDMIMOUNT_2_DRVHOSTBASE(pInterface);
521 RTCritSectEnter(&pThis->CritSect);
522
523 int rc = VINF_SUCCESS;
524 if (!pThis->fLocked)
525 {
526 if (pThis->pfnDoLock)
527 rc = pThis->pfnDoLock(pThis, true);
528 if (RT_SUCCESS(rc))
529 pThis->fLocked = true;
530 }
531 else
532 LogFlow(("%s-%d: drvHostBaseLock: already locked\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
533
534 RTCritSectLeave(&pThis->CritSect);
535 LogFlow(("%s-%d: drvHostBaseLock: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
536 return rc;
537}
538
539
540/** @copydoc PDMIMOUNT::pfnIsLocked */
541static DECLCALLBACK(int) drvHostBaseUnlock(PPDMIMOUNT pInterface)
542{
543 PDRVHOSTBASE pThis = PDMIMOUNT_2_DRVHOSTBASE(pInterface);
544 RTCritSectEnter(&pThis->CritSect);
545
546 int rc = VINF_SUCCESS;
547 if (pThis->fLocked)
548 {
549 if (pThis->pfnDoLock)
550 rc = pThis->pfnDoLock(pThis, false);
551 if (RT_SUCCESS(rc))
552 pThis->fLocked = false;
553 }
554 else
555 LogFlow(("%s-%d: drvHostBaseUnlock: not locked\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
556
557 RTCritSectLeave(&pThis->CritSect);
558 LogFlow(("%s-%d: drvHostBaseUnlock: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
559 return rc;
560}
561
562
563/** @copydoc PDMIMOUNT::pfnIsLocked */
564static DECLCALLBACK(bool) drvHostBaseIsLocked(PPDMIMOUNT pInterface)
565{
566 PDRVHOSTBASE pThis = PDMIMOUNT_2_DRVHOSTBASE(pInterface);
567 RTCritSectEnter(&pThis->CritSect);
568
569 bool fRc = pThis->fLocked;
570
571 RTCritSectLeave(&pThis->CritSect);
572 return fRc;
573}
574
575
576/* -=-=-=-=- IBase -=-=-=-=- */
577
578/**
579 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
580 */
581static DECLCALLBACK(void *) drvHostBaseQueryInterface(PPDMIBASE pInterface, const char *pszIID)
582{
583 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
584 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
585
586 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
587 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBLOCK, &pThis->IBlock);
588 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBLOCKBIOS, pThis->fBiosVisible ? &pThis->IBlockBios : NULL);
589 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMOUNT, &pThis->IMount);
590 return NULL;
591}
592
593
594/* -=-=-=-=- poller thread -=-=-=-=- */
595
596#ifdef RT_OS_DARWIN
597/** The runloop input source name for the disk arbitration events. */
598# define MY_RUN_LOOP_MODE CFSTR("drvHostBaseDA") /** @todo r=bird: Check if this will cause trouble in the same way that the one in the USB code did. */
599
600/**
601 * Gets the BSD Name (/dev/disc[0-9]+) for the service.
602 *
603 * This is done by recursing down the I/O registry until we hit upon an entry
604 * with a BSD Name. Usually we find it two levels down. (Further down under
605 * the IOCDPartitionScheme, the volume (slices) BSD Name is found. We don't
606 * seem to have to go this far fortunately.)
607 *
608 * @return VINF_SUCCESS if found, VERR_FILE_NOT_FOUND otherwise.
609 * @param Entry The current I/O registry entry reference.
610 * @param pszName Where to store the name. 128 bytes.
611 * @param cRecursions Number of recursions. This is used as an precaution
612 * just to limit the depth and avoid blowing the stack
613 * should we hit a bug or something.
614 */
615static int drvHostBaseGetBSDName(io_registry_entry_t Entry, char *pszName, unsigned cRecursions)
616{
617 int rc = VERR_FILE_NOT_FOUND;
618 io_iterator_t Children = 0;
619 kern_return_t krc = IORegistryEntryGetChildIterator(Entry, kIOServicePlane, &Children);
620 if (krc == KERN_SUCCESS)
621 {
622 io_object_t Child;
623 while ( rc == VERR_FILE_NOT_FOUND
624 && (Child = IOIteratorNext(Children)) != 0)
625 {
626 CFStringRef BSDNameStrRef = (CFStringRef)IORegistryEntryCreateCFProperty(Child, CFSTR(kIOBSDNameKey), kCFAllocatorDefault, 0);
627 if (BSDNameStrRef)
628 {
629 if (CFStringGetCString(BSDNameStrRef, pszName, 128, kCFStringEncodingUTF8))
630 rc = VINF_SUCCESS;
631 else
632 AssertFailed();
633 CFRelease(BSDNameStrRef);
634 }
635 if (rc == VERR_FILE_NOT_FOUND && cRecursions < 10)
636 rc = drvHostBaseGetBSDName(Child, pszName, cRecursions + 1);
637 IOObjectRelease(Child);
638 }
639 IOObjectRelease(Children);
640 }
641 return rc;
642}
643
644
645/**
646 * Callback notifying us that the async DADiskClaim()/DADiskUnmount call has completed.
647 *
648 * @param DiskRef The disk that was attempted claimed / unmounted.
649 * @param DissenterRef NULL on success, contains details on failure.
650 * @param pvContext Pointer to the return code variable.
651 */
652static void drvHostBaseDADoneCallback(DADiskRef DiskRef, DADissenterRef DissenterRef, void *pvContext)
653{
654 int *prc = (int *)pvContext;
655 if (!DissenterRef)
656 *prc = 0;
657 else
658 *prc = DADissenterGetStatus(DissenterRef) ? DADissenterGetStatus(DissenterRef) : -1;
659 CFRunLoopStop(CFRunLoopGetCurrent());
660}
661
662
663/**
664 * Obtain exclusive access to the DVD device, umount it if necessary.
665 *
666 * @return VBox status code.
667 * @param pThis The driver instance.
668 * @param DVDService The DVD service object.
669 */
670static int drvHostBaseObtainExclusiveAccess(PDRVHOSTBASE pThis, io_object_t DVDService)
671{
672 PPDMDRVINS pDrvIns = pThis->pDrvIns; NOREF(pDrvIns);
673
674 for (unsigned iTry = 0;; iTry++)
675 {
676 IOReturn irc = (*pThis->ppScsiTaskDI)->ObtainExclusiveAccess(pThis->ppScsiTaskDI);
677 if (irc == kIOReturnSuccess)
678 {
679 /*
680 * This is a bit weird, but if we unmounted the DVD drive we also need to
681 * unlock it afterwards or the guest won't be able to eject it later on.
682 */
683 if (pThis->pDADisk)
684 {
685 uint8_t abCmd[16] =
686 {
687 SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL, 0, 0, 0, false, 0,
688 0,0,0,0,0,0,0,0,0,0
689 };
690 DRVHostBaseScsiCmd(pThis, abCmd, 6, PDMBLOCKTXDIR_NONE, NULL, NULL, NULL, 0, 0);
691 }
692 return VINF_SUCCESS;
693 }
694 if (irc == kIOReturnExclusiveAccess)
695 return VERR_SHARING_VIOLATION; /* already used exclusivly. */
696 if (irc != kIOReturnBusy)
697 return VERR_GENERAL_FAILURE; /* not mounted */
698
699 /*
700 * Attempt to the unmount all volumes of the device.
701 * It seems we can can do this all in one go without having to enumerate the
702 * volumes (sessions) and deal with them one by one. This is very fortuitous
703 * as the disk arbitration API is a bit cumbersome to deal with.
704 */
705 if (iTry > 2)
706 return VERR_DRIVE_LOCKED;
707 char szName[128];
708 int rc = drvHostBaseGetBSDName(DVDService, &szName[0], 0);
709 if (RT_SUCCESS(rc))
710 {
711 pThis->pDASession = DASessionCreate(kCFAllocatorDefault);
712 if (pThis->pDASession)
713 {
714 DASessionScheduleWithRunLoop(pThis->pDASession, CFRunLoopGetCurrent(), MY_RUN_LOOP_MODE);
715 pThis->pDADisk = DADiskCreateFromBSDName(kCFAllocatorDefault, pThis->pDASession, szName);
716 if (pThis->pDADisk)
717 {
718 /*
719 * Try claim the device.
720 */
721 Log(("%s-%d: calling DADiskClaim on '%s'.\n", pDrvIns->pReg->szName, pDrvIns->iInstance, szName));
722 int rcDA = -2;
723 DADiskClaim(pThis->pDADisk, kDADiskClaimOptionDefault, NULL, NULL, drvHostBaseDADoneCallback, &rcDA);
724 SInt32 rc32 = CFRunLoopRunInMode(MY_RUN_LOOP_MODE, 120.0, FALSE);
725 AssertMsg(rc32 == kCFRunLoopRunStopped, ("rc32=%RI32 (%RX32)\n", rc32, rc32));
726 if ( rc32 == kCFRunLoopRunStopped
727 && !rcDA)
728 {
729 /*
730 * Try unmount the device.
731 */
732 Log(("%s-%d: calling DADiskUnmount on '%s'.\n", pDrvIns->pReg->szName, pDrvIns->iInstance, szName));
733 rcDA = -2;
734 DADiskUnmount(pThis->pDADisk, kDADiskUnmountOptionWhole, drvHostBaseDADoneCallback, &rcDA);
735 rc32 = CFRunLoopRunInMode(MY_RUN_LOOP_MODE, 120.0, FALSE);
736 AssertMsg(rc32 == kCFRunLoopRunStopped, ("rc32=%RI32 (%RX32)\n", rc32, rc32));
737 if ( rc32 == kCFRunLoopRunStopped
738 && !rcDA)
739 {
740 iTry = 99;
741 DASessionUnscheduleFromRunLoop(pThis->pDASession, CFRunLoopGetCurrent(), MY_RUN_LOOP_MODE);
742 Log(("%s-%d: unmount succeed - retrying.\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
743 continue;
744 }
745 Log(("%s-%d: umount => rc32=%d & rcDA=%#x\n", pDrvIns->pReg->szName, pDrvIns->iInstance, rc32, rcDA));
746
747 /* failed - cleanup */
748 DADiskUnclaim(pThis->pDADisk);
749 }
750 else
751 Log(("%s-%d: claim => rc32=%d & rcDA=%#x\n", pDrvIns->pReg->szName, pDrvIns->iInstance, rc32, rcDA));
752
753 CFRelease(pThis->pDADisk);
754 pThis->pDADisk = NULL;
755 }
756 else
757 Log(("%s-%d: failed to open disk '%s'!\n", pDrvIns->pReg->szName, pDrvIns->iInstance, szName));
758
759 DASessionUnscheduleFromRunLoop(pThis->pDASession, CFRunLoopGetCurrent(), MY_RUN_LOOP_MODE);
760 CFRelease(pThis->pDASession);
761 pThis->pDASession = NULL;
762 }
763 else
764 Log(("%s-%d: failed to create DA session!\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
765 }
766 RTThreadSleep(10);
767 }
768}
769#endif /* RT_OS_DARWIN */
770
771
772#ifndef RT_OS_SOLARIS
773/**
774 * Wrapper for open / RTFileOpen / IOKit.
775 *
776 * @remark The Darwin code must correspond exactly to the enumeration
777 * done in Main/darwin/iokit.c.
778 */
779static int drvHostBaseOpen(PDRVHOSTBASE pThis, PRTFILE pFileDevice, bool fReadOnly)
780{
781# ifdef RT_OS_DARWIN
782 /* Darwin is kind of special... */
783 Assert(!pFileDevice); NOREF(pFileDevice);
784 Assert(!pThis->cbBlock);
785 Assert(!pThis->MasterPort);
786 Assert(!pThis->ppMMCDI);
787 Assert(!pThis->ppScsiTaskDI);
788
789 /*
790 * Open the master port on the first invocation.
791 */
792 kern_return_t krc = IOMasterPort(MACH_PORT_NULL, &pThis->MasterPort);
793 AssertReturn(krc == KERN_SUCCESS, VERR_GENERAL_FAILURE);
794
795 /*
796 * Create a matching dictionary for searching for CD, DVD and BlueRay services in the IOKit.
797 *
798 * The idea is to find all the devices which are of class IOCDBlockStorageDevice.
799 * CD devices are represented by IOCDBlockStorageDevice class itself, while DVD and BlueRay ones
800 * have it as a parent class.
801 */
802 CFMutableDictionaryRef RefMatchingDict = IOServiceMatching("IOCDBlockStorageDevice");
803 AssertReturn(RefMatchingDict, NULL);
804
805 /*
806 * do the search and get a collection of keyboards.
807 */
808 io_iterator_t DVDServices = NULL;
809 IOReturn irc = IOServiceGetMatchingServices(pThis->MasterPort, RefMatchingDict, &DVDServices);
810 AssertMsgReturn(irc == kIOReturnSuccess, ("irc=%d\n", irc), NULL);
811 RefMatchingDict = NULL; /* the reference is consumed by IOServiceGetMatchingServices. */
812
813 /*
814 * Enumerate the matching drives (services).
815 * (This enumeration must be identical to the one performed in Main/src-server/darwin/iokit.cpp.)
816 */
817 int rc = VERR_FILE_NOT_FOUND;
818 unsigned i = 0;
819 io_object_t DVDService;
820 while ((DVDService = IOIteratorNext(DVDServices)) != 0)
821 {
822 /*
823 * Get the properties we use to identify the DVD drive.
824 *
825 * While there is a (weird 12 byte) GUID, it isn't persistent
826 * across boots. So, we have to use a combination of the
827 * vendor name and product name properties with an optional
828 * sequence number for identification.
829 */
830 CFMutableDictionaryRef PropsRef = 0;
831 krc = IORegistryEntryCreateCFProperties(DVDService, &PropsRef, kCFAllocatorDefault, kNilOptions);
832 if (krc == KERN_SUCCESS)
833 {
834 /* Get the Device Characteristics dictionary. */
835 CFDictionaryRef DevCharRef = (CFDictionaryRef)CFDictionaryGetValue(PropsRef, CFSTR(kIOPropertyDeviceCharacteristicsKey));
836 if (DevCharRef)
837 {
838 /* The vendor name. */
839 char szVendor[128];
840 char *pszVendor = &szVendor[0];
841 CFTypeRef ValueRef = CFDictionaryGetValue(DevCharRef, CFSTR(kIOPropertyVendorNameKey));
842 if ( ValueRef
843 && CFGetTypeID(ValueRef) == CFStringGetTypeID()
844 && CFStringGetCString((CFStringRef)ValueRef, szVendor, sizeof(szVendor), kCFStringEncodingUTF8))
845 pszVendor = RTStrStrip(szVendor);
846 else
847 *pszVendor = '\0';
848
849 /* The product name. */
850 char szProduct[128];
851 char *pszProduct = &szProduct[0];
852 ValueRef = CFDictionaryGetValue(DevCharRef, CFSTR(kIOPropertyProductNameKey));
853 if ( ValueRef
854 && CFGetTypeID(ValueRef) == CFStringGetTypeID()
855 && CFStringGetCString((CFStringRef)ValueRef, szProduct, sizeof(szProduct), kCFStringEncodingUTF8))
856 pszProduct = RTStrStrip(szProduct);
857 else
858 *pszProduct = '\0';
859
860 /* Construct the two names and compare thwm with the one we're searching for. */
861 char szName1[256 + 32];
862 char szName2[256 + 32];
863 if (*pszVendor || *pszProduct)
864 {
865 if (*pszVendor && *pszProduct)
866 {
867 RTStrPrintf(szName1, sizeof(szName1), "%s %s", pszVendor, pszProduct);
868 RTStrPrintf(szName2, sizeof(szName2), "%s %s (#%u)", pszVendor, pszProduct, i);
869 }
870 else
871 {
872 strcpy(szName1, *pszVendor ? pszVendor : pszProduct);
873 RTStrPrintf(szName2, sizeof(szName2), "%s (#%u)", *pszVendor ? pszVendor : pszProduct, i);
874 }
875 }
876 else
877 {
878 RTStrPrintf(szName1, sizeof(szName1), "(#%u)", i);
879 strcpy(szName2, szName1);
880 }
881
882 if ( !strcmp(szName1, pThis->pszDeviceOpen)
883 || !strcmp(szName2, pThis->pszDeviceOpen))
884 {
885 /*
886 * Found it! Now, get the client interface and stuff.
887 * Note that we could also query kIOSCSITaskDeviceUserClientTypeID here if the
888 * MMC client plugin is missing. For now we assume this won't be necessary.
889 */
890 SInt32 Score = 0;
891 IOCFPlugInInterface **ppPlugInInterface = NULL;
892 krc = IOCreatePlugInInterfaceForService(DVDService, kIOMMCDeviceUserClientTypeID, kIOCFPlugInInterfaceID,
893 &ppPlugInInterface, &Score);
894 if (krc == KERN_SUCCESS)
895 {
896 HRESULT hrc = (*ppPlugInInterface)->QueryInterface(ppPlugInInterface,
897 CFUUIDGetUUIDBytes(kIOMMCDeviceInterfaceID),
898 (LPVOID *)&pThis->ppMMCDI);
899 (*ppPlugInInterface)->Release(ppPlugInInterface);
900 ppPlugInInterface = NULL;
901 if (hrc == S_OK)
902 {
903 pThis->ppScsiTaskDI = (*pThis->ppMMCDI)->GetSCSITaskDeviceInterface(pThis->ppMMCDI);
904 if (pThis->ppScsiTaskDI)
905 rc = VINF_SUCCESS;
906 else
907 {
908 LogRel(("GetSCSITaskDeviceInterface failed on '%s'\n", pThis->pszDeviceOpen));
909 rc = VERR_NOT_SUPPORTED;
910 (*pThis->ppMMCDI)->Release(pThis->ppMMCDI);
911 }
912 }
913 else
914 {
915 rc = VERR_GENERAL_FAILURE;//RTErrConvertFromDarwinCOM(krc);
916 pThis->ppMMCDI = NULL;
917 }
918 }
919 else /* Check for kIOSCSITaskDeviceUserClientTypeID? */
920 rc = VERR_GENERAL_FAILURE;//RTErrConvertFromDarwinKern(krc);
921
922 /* Obtain exclusive access to the device so we can send SCSI commands. */
923 if (RT_SUCCESS(rc))
924 rc = drvHostBaseObtainExclusiveAccess(pThis, DVDService);
925
926 /* Cleanup on failure. */
927 if (RT_FAILURE(rc))
928 {
929 if (pThis->ppScsiTaskDI)
930 {
931 (*pThis->ppScsiTaskDI)->Release(pThis->ppScsiTaskDI);
932 pThis->ppScsiTaskDI = NULL;
933 }
934 if (pThis->ppMMCDI)
935 {
936 (*pThis->ppMMCDI)->Release(pThis->ppMMCDI);
937 pThis->ppMMCDI = NULL;
938 }
939 }
940
941 IOObjectRelease(DVDService);
942 break;
943 }
944 }
945 CFRelease(PropsRef);
946 }
947 else
948 AssertMsgFailed(("krc=%#x\n", krc));
949
950 IOObjectRelease(DVDService);
951 i++;
952 }
953
954 IOObjectRelease(DVDServices);
955 return rc;
956
957#elif defined(RT_OS_FREEBSD)
958 RTFILE hFileDevice;
959 int rc = RTFileOpen(&hFileDevice, pThis->pszDeviceOpen, RTFILE_O_READWRITE | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
960 if (RT_FAILURE(rc))
961 return rc;
962
963 /*
964 * The current device handle can't passthrough SCSI commands.
965 * We have to get he passthrough device path and open this.
966 */
967 union ccb DeviceCCB;
968 memset(&DeviceCCB, 0, sizeof(DeviceCCB));
969
970 DeviceCCB.ccb_h.func_code = XPT_GDEVLIST;
971 int rcBSD = ioctl(RTFileToNative(hFileDevice), CAMGETPASSTHRU, &DeviceCCB);
972 if (!rcBSD)
973 {
974 char *pszPassthroughDevice = NULL;
975 rc = RTStrAPrintf(&pszPassthroughDevice, "/dev/%s%u",
976 DeviceCCB.cgdl.periph_name, DeviceCCB.cgdl.unit_number);
977 if (rc >= 0)
978 {
979 RTFILE hPassthroughDevice;
980 rc = RTFileOpen(&hPassthroughDevice, pszPassthroughDevice, RTFILE_O_READWRITE | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
981 RTStrFree(pszPassthroughDevice);
982 if (RT_SUCCESS(rc))
983 {
984 /* Get needed device parameters. */
985
986 /*
987 * The device path, target id and lun id. Those are
988 * needed for the SCSI passthrough ioctl.
989 */
990 memset(&DeviceCCB, 0, sizeof(DeviceCCB));
991 DeviceCCB.ccb_h.func_code = XPT_GDEVLIST;
992
993 rcBSD = ioctl(RTFileToNative(hPassthroughDevice), CAMGETPASSTHRU, &DeviceCCB);
994 if (!rcBSD)
995 {
996 if (DeviceCCB.cgdl.status != CAM_GDEVLIST_ERROR)
997 {
998 pThis->ScsiBus = DeviceCCB.ccb_h.path_id;
999 pThis->ScsiTargetID = DeviceCCB.ccb_h.target_id;
1000 pThis->ScsiLunID = DeviceCCB.ccb_h.target_lun;
1001 *pFileDevice = hPassthroughDevice;
1002 }
1003 else
1004 {
1005 /* The passthrough device wasn't found. */
1006 rc = VERR_NOT_FOUND;
1007 }
1008 }
1009 else
1010 rc = RTErrConvertFromErrno(errno);
1011
1012 if (RT_FAILURE(rc))
1013 RTFileClose(hPassthroughDevice);
1014 }
1015 }
1016 else
1017 rc = VERR_NO_STR_MEMORY;
1018 }
1019 else
1020 rc = RTErrConvertFromErrno(errno);
1021
1022 RTFileClose(hFileDevice);
1023 return rc;
1024
1025#else
1026 uint32_t fFlags = (fReadOnly ? RTFILE_O_READ : RTFILE_O_READWRITE) | RTFILE_O_OPEN | RTFILE_O_DENY_NONE;
1027# ifdef RT_OS_LINUX
1028 fFlags |= RTFILE_O_NON_BLOCK;
1029# endif
1030 return RTFileOpen(pFileDevice, pThis->pszDeviceOpen, fFlags);
1031#endif
1032}
1033
1034#else /* RT_OS_SOLARIS */
1035
1036/**
1037 * Solaris wrapper for RTFileOpen.
1038 *
1039 * Solaris has to deal with two filehandles, a block and a raw one. Rather than messing
1040 * with drvHostBaseOpen's function signature & body, having a separate one is better.
1041 *
1042 * @returns VBox status code.
1043 */
1044static int drvHostBaseOpen(PDRVHOSTBASE pThis, PRTFILE pFileBlockDevice, PRTFILE pFileRawDevice, bool fReadOnly)
1045{
1046 unsigned fFlags = (fReadOnly ? RTFILE_O_READ : RTFILE_O_READWRITE)
1047 | RTFILE_O_OPEN | RTFILE_O_DENY_NONE | RTFILE_O_NON_BLOCK;
1048 int rc = RTFileOpen(pFileBlockDevice, pThis->pszDeviceOpen, fFlags);
1049 if (RT_SUCCESS(rc))
1050 {
1051 rc = RTFileOpen(pFileRawDevice, pThis->pszRawDeviceOpen, fFlags);
1052 if (RT_SUCCESS(rc))
1053 return rc;
1054
1055 LogRel(("DVD: failed to open device %s rc=%Rrc\n", pThis->pszRawDeviceOpen, rc));
1056 RTFileClose(*pFileBlockDevice);
1057 }
1058 else
1059 LogRel(("DVD: failed to open device %s rc=%Rrc\n", pThis->pszDeviceOpen, rc));
1060 return rc;
1061}
1062#endif /* RT_OS_SOLARIS */
1063
1064
1065/**
1066 * (Re)opens the device.
1067 *
1068 * This is used to open the device during construction, but it's also used to re-open
1069 * the device when a media is inserted. This re-open will kill off any cached data
1070 * that Linux for some peculiar reason thinks should survive a media change...
1071 *
1072 * @returns VBOX status code.
1073 * @param pThis Instance data.
1074 */
1075static int drvHostBaseReopen(PDRVHOSTBASE pThis)
1076{
1077#ifndef RT_OS_DARWIN /* Only *one* open for darwin. */
1078 LogFlow(("%s-%d: drvHostBaseReopen: '%s'\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDeviceOpen));
1079
1080 RTFILE hFileDevice;
1081#ifdef RT_OS_SOLARIS
1082 if (pThis->hFileRawDevice != NIL_RTFILE)
1083 {
1084 RTFileClose(pThis->hFileRawDevice);
1085 pThis->hFileRawDevice = NIL_RTFILE;
1086 }
1087 if (pThis->hFileDevice != NIL_RTFILE)
1088 {
1089 RTFileClose(pThis->hFileDevice);
1090 pThis->hFileDevice = NIL_RTFILE;
1091 }
1092 RTFILE hFileRawDevice;
1093 int rc = drvHostBaseOpen(pThis, &hFileDevice, &hFileRawDevice, pThis->fReadOnlyConfig);
1094#else
1095 int rc = drvHostBaseOpen(pThis, &hFileDevice, pThis->fReadOnlyConfig);
1096#endif
1097 if (RT_FAILURE(rc))
1098 {
1099 if (!pThis->fReadOnlyConfig)
1100 {
1101 LogFlow(("%s-%d: drvHostBaseReopen: '%s' - retry readonly (%Rrc)\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDeviceOpen, rc));
1102#ifdef RT_OS_SOLARIS
1103 rc = drvHostBaseOpen(pThis, &hFileDevice, &hFileRawDevice, false);
1104#else
1105 rc = drvHostBaseOpen(pThis, &hFileDevice, false);
1106#endif
1107 }
1108 if (RT_FAILURE(rc))
1109 {
1110 LogFlow(("%s-%d: failed to open device '%s', rc=%Rrc\n",
1111 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDevice, rc));
1112 return rc;
1113 }
1114 pThis->fReadOnly = true;
1115 }
1116 else
1117 pThis->fReadOnly = pThis->fReadOnlyConfig;
1118
1119#ifdef RT_OS_SOLARIS
1120 if (pThis->hFileRawDevice != NIL_RTFILE)
1121 RTFileClose(pThis->hFileRawDevice);
1122 pThis->hFileRawDevice = hFileRawDevice;
1123#endif
1124
1125 if (pThis->hFileDevice != NIL_RTFILE)
1126 RTFileClose(pThis->hFileDevice);
1127 pThis->hFileDevice = hFileDevice;
1128#endif /* !RT_OS_DARWIN */
1129 return VINF_SUCCESS;
1130}
1131
1132
1133/**
1134 * Queries the media size.
1135 *
1136 * @returns VBox status code.
1137 * @param pThis Pointer to the instance data.
1138 * @param pcb Where to store the media size in bytes.
1139 */
1140static int drvHostBaseGetMediaSize(PDRVHOSTBASE pThis, uint64_t *pcb)
1141{
1142#if defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
1143 /*
1144 * Try a READ_CAPACITY command...
1145 */
1146 struct
1147 {
1148 uint32_t cBlocks;
1149 uint32_t cbBlock;
1150 } Buf = {0, 0};
1151 uint32_t cbBuf = sizeof(Buf);
1152 uint8_t abCmd[16] =
1153 {
1154 SCSI_READ_CAPACITY, 0, 0, 0, 0, 0, 0,
1155 0,0,0,0,0,0,0,0,0
1156 };
1157 int rc = DRVHostBaseScsiCmd(pThis, abCmd, 6, PDMBLOCKTXDIR_FROM_DEVICE, &Buf, &cbBuf, NULL, 0, 0);
1158 if (RT_SUCCESS(rc))
1159 {
1160 Assert(cbBuf == sizeof(Buf));
1161 Buf.cBlocks = RT_BE2H_U32(Buf.cBlocks);
1162 Buf.cbBlock = RT_BE2H_U32(Buf.cbBlock);
1163 //if (Buf.cbBlock > 2048) /* everyone else is doing this... check if it needed/right.*/
1164 // Buf.cbBlock = 2048;
1165 pThis->cbBlock = Buf.cbBlock;
1166
1167 *pcb = (uint64_t)Buf.cBlocks * Buf.cbBlock;
1168 }
1169 return rc;
1170
1171#elif defined(RT_OS_SOLARIS)
1172 /*
1173 * Sun docs suggests using DKIOCGGEOM instead of DKIOCGMEDIAINFO, but
1174 * Sun themselves use DKIOCGMEDIAINFO for DVDs/CDs, and use DKIOCGGEOM
1175 * for secondary storage devices.
1176 */
1177 struct dk_minfo MediaInfo;
1178 if (ioctl(RTFileToNative(pThis->hFileRawDevice), DKIOCGMEDIAINFO, &MediaInfo) == 0)
1179 {
1180 *pcb = MediaInfo.dki_capacity * (uint64_t)MediaInfo.dki_lbsize;
1181 return VINF_SUCCESS;
1182 }
1183 return RTFileSeek(pThis->hFileDevice, 0, RTFILE_SEEK_END, pcb);
1184
1185#elif defined(RT_OS_WINDOWS)
1186 /* use NT api, retry a few times if the media is being verified. */
1187 IO_STATUS_BLOCK IoStatusBlock = {0};
1188 FILE_FS_SIZE_INFORMATION FsSize= {0};
1189 NTSTATUS rcNt = NtQueryVolumeInformationFile((HANDLE)RTFileToNative(pThis->hFileDevice), &IoStatusBlock,
1190 &FsSize, sizeof(FsSize), FileFsSizeInformation);
1191 int cRetries = 5;
1192 while (rcNt == STATUS_VERIFY_REQUIRED && cRetries-- > 0)
1193 {
1194 RTThreadSleep(10);
1195 rcNt = NtQueryVolumeInformationFile((HANDLE)RTFileToNative(pThis->hFileDevice), &IoStatusBlock,
1196 &FsSize, sizeof(FsSize), FileFsSizeInformation);
1197 }
1198 if (rcNt >= 0)
1199 {
1200 *pcb = FsSize.TotalAllocationUnits.QuadPart * FsSize.BytesPerSector;
1201 return VINF_SUCCESS;
1202 }
1203
1204 /* convert nt status code to VBox status code. */
1205 /** @todo Make conversion function!. */
1206 int rc = VERR_GENERAL_FAILURE;
1207 switch (rcNt)
1208 {
1209 case STATUS_NO_MEDIA_IN_DEVICE: rc = VERR_MEDIA_NOT_PRESENT; break;
1210 case STATUS_VERIFY_REQUIRED: rc = VERR_TRY_AGAIN; break;
1211 }
1212 LogFlow(("drvHostBaseGetMediaSize: NtQueryVolumeInformationFile -> %#lx\n", rcNt, rc));
1213 return rc;
1214#else
1215 return RTFileSeek(pThis->hFileDevice, 0, RTFILE_SEEK_END, pcb);
1216#endif
1217}
1218
1219
1220#if defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
1221/**
1222 * Execute a SCSI command.
1223 *
1224 * @param pThis The instance data.
1225 * @param pbCmd Pointer to the SCSI command.
1226 * @param cbCmd The size of the SCSI command.
1227 * @param enmTxDir The transfer direction.
1228 * @param pvBuf The buffer. Can be NULL if enmTxDir is PDMBLOCKTXDIR_NONE.
1229 * @param pcbBuf Where to get the buffer size from and put the actual transfer size. Can be NULL.
1230 * @param pbSense Where to put the sense data. Can be NULL.
1231 * @param cbSense Size of the sense data buffer.
1232 * @param cTimeoutMillies The timeout. 0 mean the default timeout.
1233 *
1234 * @returns VINF_SUCCESS on success (no sense code).
1235 * @returns VERR_UNRESOLVED_ERROR if sense code is present.
1236 * @returns Some other VBox status code on failures without sense code.
1237 *
1238 * @todo Fix VERR_UNRESOLVED_ERROR abuse.
1239 */
1240DECLCALLBACK(int) DRVHostBaseScsiCmd(PDRVHOSTBASE pThis, const uint8_t *pbCmd, size_t cbCmd, PDMBLOCKTXDIR enmTxDir,
1241 void *pvBuf, uint32_t *pcbBuf, uint8_t *pbSense, size_t cbSense, uint32_t cTimeoutMillies)
1242{
1243 /*
1244 * Minimal input validation.
1245 */
1246 Assert(enmTxDir == PDMBLOCKTXDIR_NONE || enmTxDir == PDMBLOCKTXDIR_FROM_DEVICE || enmTxDir == PDMBLOCKTXDIR_TO_DEVICE);
1247 Assert(!pvBuf || pcbBuf);
1248 Assert(pvBuf || enmTxDir == PDMBLOCKTXDIR_NONE);
1249 Assert(pbSense || !cbSense);
1250 AssertPtr(pbCmd);
1251 Assert(cbCmd <= 16 && cbCmd >= 1);
1252 const uint32_t cbBuf = pcbBuf ? *pcbBuf : 0;
1253 if (pcbBuf)
1254 *pcbBuf = 0;
1255
1256# ifdef RT_OS_DARWIN
1257 Assert(pThis->ppScsiTaskDI);
1258
1259 int rc = VERR_GENERAL_FAILURE;
1260 SCSITaskInterface **ppScsiTaskI = (*pThis->ppScsiTaskDI)->CreateSCSITask(pThis->ppScsiTaskDI);
1261 if (!ppScsiTaskI)
1262 return VERR_NO_MEMORY;
1263 do
1264 {
1265 /* Setup the scsi command. */
1266 SCSICommandDescriptorBlock cdb = { 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 };
1267 memcpy(&cdb[0], pbCmd, cbCmd);
1268 IOReturn irc = (*ppScsiTaskI)->SetCommandDescriptorBlock(ppScsiTaskI, cdb, cbCmd);
1269 AssertBreak(irc == kIOReturnSuccess);
1270
1271 /* Setup the buffer. */
1272 if (enmTxDir == PDMBLOCKTXDIR_NONE)
1273 irc = (*ppScsiTaskI)->SetScatterGatherEntries(ppScsiTaskI, NULL, 0, 0, kSCSIDataTransfer_NoDataTransfer);
1274 else
1275 {
1276 IOVirtualRange Range = { (IOVirtualAddress)pvBuf, cbBuf };
1277 irc = (*ppScsiTaskI)->SetScatterGatherEntries(ppScsiTaskI, &Range, 1, cbBuf,
1278 enmTxDir == PDMBLOCKTXDIR_FROM_DEVICE
1279 ? kSCSIDataTransfer_FromTargetToInitiator
1280 : kSCSIDataTransfer_FromInitiatorToTarget);
1281 }
1282 AssertBreak(irc == kIOReturnSuccess);
1283
1284 /* Set the timeout. */
1285 irc = (*ppScsiTaskI)->SetTimeoutDuration(ppScsiTaskI, cTimeoutMillies ? cTimeoutMillies : 30000 /*ms*/);
1286 AssertBreak(irc == kIOReturnSuccess);
1287
1288 /* Execute the command and get the response. */
1289 SCSI_Sense_Data SenseData = { 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 };
1290 SCSIServiceResponse ServiceResponse = kSCSIServiceResponse_Request_In_Process;
1291 SCSITaskStatus TaskStatus = kSCSITaskStatus_GOOD;
1292 UInt64 cbReturned = 0;
1293 irc = (*ppScsiTaskI)->ExecuteTaskSync(ppScsiTaskI, &SenseData, &TaskStatus, &cbReturned);
1294 AssertBreak(irc == kIOReturnSuccess);
1295 if (pcbBuf)
1296 *pcbBuf = (int32_t)cbReturned;
1297
1298 irc = (*ppScsiTaskI)->GetSCSIServiceResponse(ppScsiTaskI, &ServiceResponse);
1299 AssertBreak(irc == kIOReturnSuccess);
1300 AssertBreak(ServiceResponse == kSCSIServiceResponse_TASK_COMPLETE);
1301
1302 if (TaskStatus == kSCSITaskStatus_GOOD)
1303 rc = VINF_SUCCESS;
1304 else if ( TaskStatus == kSCSITaskStatus_CHECK_CONDITION
1305 && pbSense)
1306 {
1307 memset(pbSense, 0, cbSense); /* lazy */
1308 memcpy(pbSense, &SenseData, RT_MIN(sizeof(SenseData), cbSense));
1309 rc = VERR_UNRESOLVED_ERROR;
1310 }
1311 /** @todo convert sense codes when caller doesn't wish to do this himself. */
1312 /*else if ( TaskStatus == kSCSITaskStatus_CHECK_CONDITION
1313 && SenseData.ADDITIONAL_SENSE_CODE == 0x3A)
1314 rc = VERR_MEDIA_NOT_PRESENT; */
1315 else
1316 {
1317 rc = enmTxDir == PDMBLOCKTXDIR_NONE
1318 ? VERR_DEV_IO_ERROR
1319 : enmTxDir == PDMBLOCKTXDIR_FROM_DEVICE
1320 ? VERR_READ_ERROR
1321 : VERR_WRITE_ERROR;
1322 if (pThis->cLogRelErrors++ < 10)
1323 LogRel(("DVD scsi error: cmd={%.*Rhxs} TaskStatus=%#x key=%#x ASC=%#x ASCQ=%#x (%Rrc)\n",
1324 cbCmd, pbCmd, TaskStatus, SenseData.SENSE_KEY, SenseData.ADDITIONAL_SENSE_CODE,
1325 SenseData.ADDITIONAL_SENSE_CODE_QUALIFIER, rc));
1326 }
1327 } while (0);
1328
1329 (*ppScsiTaskI)->Release(ppScsiTaskI);
1330
1331# elif defined(RT_OS_FREEBSD)
1332 int rc = VINF_SUCCESS;
1333 int rcBSD = 0;
1334 union ccb DeviceCCB;
1335 union ccb *pDeviceCCB = &DeviceCCB;
1336 u_int32_t fFlags;
1337
1338 memset(pDeviceCCB, 0, sizeof(DeviceCCB));
1339 pDeviceCCB->ccb_h.path_id = pThis->ScsiBus;
1340 pDeviceCCB->ccb_h.target_id = pThis->ScsiTargetID;
1341 pDeviceCCB->ccb_h.target_lun = pThis->ScsiLunID;
1342
1343 /* The SCSI INQUIRY command can't be passed through directly. */
1344 if (pbCmd[0] == SCSI_INQUIRY)
1345 {
1346 pDeviceCCB->ccb_h.func_code = XPT_GDEV_TYPE;
1347
1348 rcBSD = ioctl(RTFileToNative(pThis->hFileDevice), CAMIOCOMMAND, pDeviceCCB);
1349 if (!rcBSD)
1350 {
1351 uint32_t cbCopy = cbBuf < sizeof(struct scsi_inquiry_data)
1352 ? cbBuf
1353 : sizeof(struct scsi_inquiry_data);;
1354 memcpy(pvBuf, &pDeviceCCB->cgd.inq_data, cbCopy);
1355 memset(pbSense, 0, cbSense);
1356
1357 if (pcbBuf)
1358 *pcbBuf = cbCopy;
1359 }
1360 else
1361 rc = RTErrConvertFromErrno(errno);
1362 }
1363 else
1364 {
1365 /* Copy the CDB. */
1366 memcpy(&pDeviceCCB->csio.cdb_io.cdb_bytes, pbCmd, cbCmd);
1367
1368 /* Set direction. */
1369 if (enmTxDir == PDMBLOCKTXDIR_NONE)
1370 fFlags = CAM_DIR_NONE;
1371 else if (enmTxDir == PDMBLOCKTXDIR_FROM_DEVICE)
1372 fFlags = CAM_DIR_IN;
1373 else
1374 fFlags = CAM_DIR_OUT;
1375
1376 fFlags |= CAM_DEV_QFRZDIS;
1377
1378 cam_fill_csio(&pDeviceCCB->csio, 1, NULL, fFlags, MSG_SIMPLE_Q_TAG,
1379 (u_int8_t *)pvBuf, cbBuf, cbSense, cbCmd,
1380 cTimeoutMillies ? cTimeoutMillies : 30000/* timeout */);
1381
1382 /* Send command */
1383 rcBSD = ioctl(RTFileToNative(pThis->hFileDevice), CAMIOCOMMAND, pDeviceCCB);
1384 if (!rcBSD)
1385 {
1386 switch (pDeviceCCB->ccb_h.status & CAM_STATUS_MASK)
1387 {
1388 case CAM_REQ_CMP:
1389 rc = VINF_SUCCESS;
1390 break;
1391 case CAM_SEL_TIMEOUT:
1392 rc = VERR_DEV_IO_ERROR;
1393 break;
1394 case CAM_CMD_TIMEOUT:
1395 rc = VERR_TIMEOUT;
1396 break;
1397 default:
1398 rc = VERR_DEV_IO_ERROR;
1399 }
1400
1401 if (pcbBuf)
1402 *pcbBuf = cbBuf - pDeviceCCB->csio.resid;
1403
1404 if (pbSense)
1405 memcpy(pbSense, &pDeviceCCB->csio.sense_data,
1406 cbSense - pDeviceCCB->csio.sense_resid);
1407 }
1408 else
1409 rc = RTErrConvertFromErrno(errno);
1410 }
1411# endif
1412
1413 return rc;
1414}
1415#endif
1416
1417
1418/**
1419 * Media present.
1420 * Query the size and notify the above driver / device.
1421 *
1422 * @param pThis The instance data.
1423 */
1424int DRVHostBaseMediaPresent(PDRVHOSTBASE pThis)
1425{
1426 /*
1427 * Open the drive.
1428 */
1429 int rc = drvHostBaseReopen(pThis);
1430 if (RT_FAILURE(rc))
1431 return rc;
1432
1433 /*
1434 * Determine the size.
1435 */
1436 uint64_t cb;
1437 rc = pThis->pfnGetMediaSize(pThis, &cb);
1438 if (RT_FAILURE(rc))
1439 {
1440 LogFlow(("%s-%d: failed to figure media size of %s, rc=%Rrc\n",
1441 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDevice, rc));
1442 return rc;
1443 }
1444
1445 /*
1446 * Update the data and inform the unit.
1447 */
1448 pThis->cbSize = cb;
1449 pThis->fMediaPresent = true;
1450 if (pThis->pDrvMountNotify)
1451 pThis->pDrvMountNotify->pfnMountNotify(pThis->pDrvMountNotify);
1452 LogFlow(("%s-%d: drvHostBaseMediaPresent: cbSize=%lld (%#llx)\n",
1453 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->cbSize, pThis->cbSize));
1454 return VINF_SUCCESS;
1455}
1456
1457
1458/**
1459 * Media no longer present.
1460 * @param pThis The instance data.
1461 */
1462void DRVHostBaseMediaNotPresent(PDRVHOSTBASE pThis)
1463{
1464 pThis->fMediaPresent = false;
1465 pThis->fLocked = false;
1466 pThis->PCHSGeometry.cCylinders = 0;
1467 pThis->PCHSGeometry.cHeads = 0;
1468 pThis->PCHSGeometry.cSectors = 0;
1469 pThis->LCHSGeometry.cCylinders = 0;
1470 pThis->LCHSGeometry.cHeads = 0;
1471 pThis->LCHSGeometry.cSectors = 0;
1472 if (pThis->pDrvMountNotify)
1473 pThis->pDrvMountNotify->pfnUnmountNotify(pThis->pDrvMountNotify);
1474}
1475
1476
1477#ifdef RT_OS_WINDOWS
1478
1479/**
1480 * Window procedure for the invisible window used to catch the WM_DEVICECHANGE broadcasts.
1481 */
1482static LRESULT CALLBACK DeviceChangeWindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
1483{
1484 Log2(("DeviceChangeWindowProc: hwnd=%08x uMsg=%08x\n", hwnd, uMsg));
1485 if (uMsg == WM_DESTROY)
1486 {
1487 PDRVHOSTBASE pThis = (PDRVHOSTBASE)GetWindowLongPtr(hwnd, GWLP_USERDATA);
1488 if (pThis)
1489 ASMAtomicXchgSize(&pThis->hwndDeviceChange, NULL);
1490 PostQuitMessage(0);
1491 }
1492
1493 if (uMsg != WM_DEVICECHANGE)
1494 return DefWindowProc(hwnd, uMsg, wParam, lParam);
1495
1496 PDEV_BROADCAST_HDR lpdb = (PDEV_BROADCAST_HDR)lParam;
1497 PDRVHOSTBASE pThis = (PDRVHOSTBASE)GetWindowLongPtr(hwnd, GWLP_USERDATA);
1498 Assert(pThis);
1499 if (pThis == NULL)
1500 return 0;
1501
1502 switch (wParam)
1503 {
1504 case DBT_DEVICEARRIVAL:
1505 case DBT_DEVICEREMOVECOMPLETE:
1506 // Check whether a CD or DVD was inserted into or removed from a drive.
1507 if (lpdb->dbch_devicetype == DBT_DEVTYP_VOLUME)
1508 {
1509 PDEV_BROADCAST_VOLUME lpdbv = (PDEV_BROADCAST_VOLUME)lpdb;
1510 if ( (lpdbv->dbcv_flags & DBTF_MEDIA)
1511 && (pThis->fUnitMask & lpdbv->dbcv_unitmask))
1512 {
1513 RTCritSectEnter(&pThis->CritSect);
1514 if (wParam == DBT_DEVICEARRIVAL)
1515 {
1516 int cRetries = 10;
1517 int rc = DRVHostBaseMediaPresent(pThis);
1518 while (RT_FAILURE(rc) && cRetries-- > 0)
1519 {
1520 RTThreadSleep(50);
1521 rc = DRVHostBaseMediaPresent(pThis);
1522 }
1523 }
1524 else
1525 DRVHostBaseMediaNotPresent(pThis);
1526 RTCritSectLeave(&pThis->CritSect);
1527 }
1528 }
1529 break;
1530 }
1531 return TRUE;
1532}
1533
1534#endif /* RT_OS_WINDOWS */
1535
1536
1537/**
1538 * This thread will periodically poll the device for media presence.
1539 *
1540 * @returns Ignored.
1541 * @param ThreadSelf Handle of this thread. Ignored.
1542 * @param pvUser Pointer to the driver instance structure.
1543 */
1544static DECLCALLBACK(int) drvHostBaseMediaThread(RTTHREAD ThreadSelf, void *pvUser)
1545{
1546 PDRVHOSTBASE pThis = (PDRVHOSTBASE)pvUser;
1547 LogFlow(("%s-%d: drvHostBaseMediaThread: ThreadSelf=%p pvUser=%p\n",
1548 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, ThreadSelf, pvUser));
1549#ifdef RT_OS_WINDOWS
1550 static WNDCLASS s_classDeviceChange = {0};
1551 static ATOM s_hAtomDeviceChange = 0;
1552
1553 /*
1554 * Register custom window class.
1555 */
1556 if (s_hAtomDeviceChange == 0)
1557 {
1558 memset(&s_classDeviceChange, 0, sizeof(s_classDeviceChange));
1559 s_classDeviceChange.lpfnWndProc = DeviceChangeWindowProc;
1560 s_classDeviceChange.lpszClassName = "VBOX_DeviceChangeClass";
1561 s_classDeviceChange.hInstance = GetModuleHandle("VBoxDD.dll");
1562 Assert(s_classDeviceChange.hInstance);
1563 s_hAtomDeviceChange = RegisterClassA(&s_classDeviceChange);
1564 Assert(s_hAtomDeviceChange);
1565 }
1566
1567 /*
1568 * Create Window w/ the pThis as user data.
1569 */
1570 HWND hwnd = CreateWindow((LPCTSTR)s_hAtomDeviceChange, "", WS_POPUP, 0, 0, 0, 0, 0, 0, s_classDeviceChange.hInstance, 0);
1571 AssertMsg(hwnd, ("CreateWindow failed with %d\n", GetLastError()));
1572 SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)pThis);
1573
1574 /*
1575 * Signal the waiting EMT thread that everything went fine.
1576 */
1577 ASMAtomicXchgSize(&pThis->hwndDeviceChange, hwnd);
1578 RTThreadUserSignal(ThreadSelf);
1579 if (!hwnd)
1580 {
1581 LogFlow(("%s-%d: drvHostBaseMediaThread: returns VERR_GENERAL_FAILURE\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
1582 return VERR_GENERAL_FAILURE;
1583 }
1584 LogFlow(("%s-%d: drvHostBaseMediaThread: Created hwndDeviceChange=%p\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, hwnd));
1585
1586 /*
1587 * Message pump.
1588 */
1589 MSG Msg;
1590 BOOL fRet;
1591 while ((fRet = GetMessage(&Msg, NULL, 0, 0)) != FALSE)
1592 {
1593 if (fRet != -1)
1594 {
1595 TranslateMessage(&Msg);
1596 DispatchMessage(&Msg);
1597 }
1598 //else: handle the error and possibly exit
1599 }
1600 Assert(!pThis->hwndDeviceChange);
1601
1602#else /* !RT_OS_WINDOWS */
1603 bool fFirst = true;
1604 int cRetries = 10;
1605 while (!pThis->fShutdownPoller)
1606 {
1607 /*
1608 * Perform the polling (unless we've run out of 50ms retries).
1609 */
1610 if ( pThis->pfnPoll
1611 && cRetries-- > 0)
1612 {
1613
1614 int rc = pThis->pfnPoll(pThis);
1615 if (RT_FAILURE(rc))
1616 {
1617 RTSemEventWait(pThis->EventPoller, 50);
1618 continue;
1619 }
1620 }
1621
1622 /*
1623 * Signal EMT after the first go.
1624 */
1625 if (fFirst)
1626 {
1627 RTThreadUserSignal(ThreadSelf);
1628 fFirst = false;
1629 }
1630
1631 /*
1632 * Sleep.
1633 */
1634 int rc = RTSemEventWait(pThis->EventPoller, pThis->cMilliesPoller);
1635 if ( RT_FAILURE(rc)
1636 && rc != VERR_TIMEOUT)
1637 {
1638 AssertMsgFailed(("rc=%Rrc\n", rc));
1639 pThis->ThreadPoller = NIL_RTTHREAD;
1640 LogFlow(("drvHostBaseMediaThread: returns %Rrc\n", rc));
1641 return rc;
1642 }
1643 cRetries = 10;
1644 }
1645
1646#endif /* !RT_OS_WINDOWS */
1647
1648 /* (Don't clear the thread handle here, the destructor thread is using it to wait.) */
1649 LogFlow(("%s-%d: drvHostBaseMediaThread: returns VINF_SUCCESS\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
1650 return VINF_SUCCESS;
1651}
1652
1653/* -=-=-=-=- driver interface -=-=-=-=- */
1654
1655
1656/**
1657 * Done state load operation.
1658 *
1659 * @returns VBox load code.
1660 * @param pDrvIns Driver instance of the driver which registered the data unit.
1661 * @param pSSM SSM operation handle.
1662 */
1663static DECLCALLBACK(int) drvHostBaseLoadDone(PPDMDRVINS pDrvIns, PSSMHANDLE pSSM)
1664{
1665 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
1666 LogFlow(("%s-%d: drvHostBaseMediaThread:\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
1667 RTCritSectEnter(&pThis->CritSect);
1668
1669 /*
1670 * Tell the device/driver above us that the media status is uncertain.
1671 */
1672 if (pThis->pDrvMountNotify)
1673 {
1674 pThis->pDrvMountNotify->pfnUnmountNotify(pThis->pDrvMountNotify);
1675 if (pThis->fMediaPresent)
1676 pThis->pDrvMountNotify->pfnMountNotify(pThis->pDrvMountNotify);
1677 }
1678
1679 RTCritSectLeave(&pThis->CritSect);
1680 return VINF_SUCCESS;
1681}
1682
1683
1684/** @copydoc FNPDMDRVDESTRUCT */
1685DECLCALLBACK(void) DRVHostBaseDestruct(PPDMDRVINS pDrvIns)
1686{
1687 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
1688 LogFlow(("%s-%d: drvHostBaseDestruct: iInstance=%d\n", pDrvIns->pReg->szName, pDrvIns->iInstance, pDrvIns->iInstance));
1689
1690 /*
1691 * Terminate the thread.
1692 */
1693 if (pThis->ThreadPoller != NIL_RTTHREAD)
1694 {
1695 pThis->fShutdownPoller = true;
1696 int rc;
1697 int cTimes = 50;
1698 do
1699 {
1700#ifdef RT_OS_WINDOWS
1701 if (pThis->hwndDeviceChange)
1702 PostMessage(pThis->hwndDeviceChange, WM_CLOSE, 0, 0); /* default win proc will destroy the window */
1703#else
1704 RTSemEventSignal(pThis->EventPoller);
1705#endif
1706 rc = RTThreadWait(pThis->ThreadPoller, 100, NULL);
1707 } while (cTimes-- > 0 && rc == VERR_TIMEOUT);
1708
1709 if (!rc)
1710 pThis->ThreadPoller = NIL_RTTHREAD;
1711 }
1712
1713 /*
1714 * Unlock the drive if we've locked it or we're in passthru mode.
1715 */
1716#ifdef RT_OS_DARWIN
1717 if ( ( pThis->fLocked
1718 || pThis->IBlock.pfnSendCmd)
1719 && pThis->ppScsiTaskDI
1720#else /** @todo Check if the other guys can mix pfnDoLock with scsi passthru.
1721 * (We're currently not unlocking the device after use. See todo in DevATA.cpp.) */
1722 if ( pThis->fLocked
1723 && pThis->hFileDevice != NIL_RTFILE
1724#endif
1725 && pThis->pfnDoLock)
1726 {
1727 int rc = pThis->pfnDoLock(pThis, false);
1728 if (RT_SUCCESS(rc))
1729 pThis->fLocked = false;
1730 }
1731
1732 /*
1733 * Cleanup the other resources.
1734 */
1735#ifdef RT_OS_WINDOWS
1736 if (pThis->hwndDeviceChange)
1737 {
1738 if (SetWindowLongPtr(pThis->hwndDeviceChange, GWLP_USERDATA, 0) == (LONG_PTR)pThis)
1739 PostMessage(pThis->hwndDeviceChange, WM_CLOSE, 0, 0); /* default win proc will destroy the window */
1740 pThis->hwndDeviceChange = NULL;
1741 }
1742#else
1743 if (pThis->EventPoller != NULL)
1744 {
1745 RTSemEventDestroy(pThis->EventPoller);
1746 pThis->EventPoller = NULL;
1747 }
1748#endif
1749
1750#ifdef RT_OS_DARWIN
1751 /*
1752 * The unclaiming doesn't seem to mean much, the DVD is actually
1753 * remounted when we release exclusive access. I'm not quite sure
1754 * if I should put the unclaim first or not...
1755 *
1756 * Anyway, that it's automatically remounted very good news for us,
1757 * because that means we don't have to mess with that ourselves. Of
1758 * course there is the unlikely scenario that we've succeeded in claiming
1759 * and umount the DVD but somehow failed to gain exclusive scsi access...
1760 */
1761 if (pThis->ppScsiTaskDI)
1762 {
1763 LogFlow(("%s-%d: releasing exclusive scsi access!\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
1764 (*pThis->ppScsiTaskDI)->ReleaseExclusiveAccess(pThis->ppScsiTaskDI);
1765 (*pThis->ppScsiTaskDI)->Release(pThis->ppScsiTaskDI);
1766 pThis->ppScsiTaskDI = NULL;
1767 }
1768 if (pThis->pDADisk)
1769 {
1770 LogFlow(("%s-%d: unclaiming the disk!\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
1771 DADiskUnclaim(pThis->pDADisk);
1772 CFRelease(pThis->pDADisk);
1773 pThis->pDADisk = NULL;
1774 }
1775 if (pThis->ppMMCDI)
1776 {
1777 LogFlow(("%s-%d: releasing the MMC object!\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
1778 (*pThis->ppMMCDI)->Release(pThis->ppMMCDI);
1779 pThis->ppMMCDI = NULL;
1780 }
1781 if (pThis->MasterPort)
1782 {
1783 mach_port_deallocate(mach_task_self(), pThis->MasterPort);
1784 pThis->MasterPort = NULL;
1785 }
1786 if (pThis->pDASession)
1787 {
1788 LogFlow(("%s-%d: releasing the DA session!\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
1789 CFRelease(pThis->pDASession);
1790 pThis->pDASession = NULL;
1791 }
1792#else
1793 if (pThis->hFileDevice != NIL_RTFILE)
1794 {
1795 int rc = RTFileClose(pThis->hFileDevice);
1796 AssertRC(rc);
1797 pThis->hFileDevice = NIL_RTFILE;
1798 }
1799#endif
1800
1801#ifdef RT_OS_SOLARIS
1802 if (pThis->hFileRawDevice != NIL_RTFILE)
1803 {
1804 int rc = RTFileClose(pThis->hFileRawDevice);
1805 AssertRC(rc);
1806 pThis->hFileRawDevice = NIL_RTFILE;
1807 }
1808
1809 if (pThis->pszRawDeviceOpen)
1810 {
1811 RTStrFree(pThis->pszRawDeviceOpen);
1812 pThis->pszRawDeviceOpen = NULL;
1813 }
1814#endif
1815
1816 if (pThis->pszDevice)
1817 {
1818 MMR3HeapFree(pThis->pszDevice);
1819 pThis->pszDevice = NULL;
1820 }
1821
1822 if (pThis->pszDeviceOpen)
1823 {
1824 RTStrFree(pThis->pszDeviceOpen);
1825 pThis->pszDeviceOpen = NULL;
1826 }
1827
1828 /* Forget about the notifications. */
1829 pThis->pDrvMountNotify = NULL;
1830
1831 /* Leave the instance operational if this is just a cleanup of the state
1832 * after an attach error happened. So don't destroy the critsect then. */
1833 if (!pThis->fKeepInstance && RTCritSectIsInitialized(&pThis->CritSect))
1834 RTCritSectDelete(&pThis->CritSect);
1835 LogFlow(("%s-%d: drvHostBaseDestruct completed\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
1836}
1837
1838
1839/**
1840 * Initializes the instance data (init part 1).
1841 *
1842 * The driver which derives from this base driver will override function pointers after
1843 * calling this method, and complete the construction by calling DRVHostBaseInitFinish().
1844 *
1845 * On failure call DRVHostBaseDestruct().
1846 *
1847 * @returns VBox status code.
1848 * @param pDrvIns Driver instance.
1849 * @param pCfg Configuration handle.
1850 * @param enmType Device type.
1851 */
1852int DRVHostBaseInitData(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, PDMBLOCKTYPE enmType)
1853{
1854 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
1855 LogFlow(("%s-%d: DRVHostBaseInitData: iInstance=%d\n", pDrvIns->pReg->szName, pDrvIns->iInstance, pDrvIns->iInstance));
1856
1857 /*
1858 * Initialize most of the data members.
1859 */
1860 pThis->pDrvIns = pDrvIns;
1861 pThis->fKeepInstance = false;
1862 pThis->ThreadPoller = NIL_RTTHREAD;
1863#ifdef RT_OS_DARWIN
1864 pThis->MasterPort = NULL;
1865 pThis->ppMMCDI = NULL;
1866 pThis->ppScsiTaskDI = NULL;
1867 pThis->cbBlock = 0;
1868 pThis->pDADisk = NULL;
1869 pThis->pDASession = NULL;
1870#else
1871 pThis->hFileDevice = NIL_RTFILE;
1872#endif
1873#ifdef RT_OS_SOLARIS
1874 pThis->hFileRawDevice = NIL_RTFILE;
1875#endif
1876 pThis->enmType = enmType;
1877 //pThis->cErrors = 0;
1878 pThis->fAttachFailError = true; /* It's an error until we've read the config. */
1879
1880 pThis->pfnGetMediaSize = drvHostBaseGetMediaSize;
1881
1882 /* IBase. */
1883 pDrvIns->IBase.pfnQueryInterface = drvHostBaseQueryInterface;
1884
1885 /* IBlock. */
1886 pThis->IBlock.pfnRead = drvHostBaseRead;
1887 pThis->IBlock.pfnWrite = drvHostBaseWrite;
1888 pThis->IBlock.pfnFlush = drvHostBaseFlush;
1889 pThis->IBlock.pfnIsReadOnly = drvHostBaseIsReadOnly;
1890 pThis->IBlock.pfnGetSize = drvHostBaseGetSize;
1891 pThis->IBlock.pfnGetType = drvHostBaseGetType;
1892 pThis->IBlock.pfnGetUuid = drvHostBaseGetUuid;
1893
1894 /* IBlockBios. */
1895 pThis->IBlockBios.pfnGetPCHSGeometry = drvHostBaseGetPCHSGeometry;
1896 pThis->IBlockBios.pfnSetPCHSGeometry = drvHostBaseSetPCHSGeometry;
1897 pThis->IBlockBios.pfnGetLCHSGeometry = drvHostBaseGetLCHSGeometry;
1898 pThis->IBlockBios.pfnSetLCHSGeometry = drvHostBaseSetLCHSGeometry;
1899 pThis->IBlockBios.pfnIsVisible = drvHostBaseIsVisible;
1900 pThis->IBlockBios.pfnGetType = drvHostBaseBiosGetType;
1901
1902 /* IMount. */
1903 pThis->IMount.pfnMount = drvHostBaseMount;
1904 pThis->IMount.pfnUnmount = drvHostBaseUnmount;
1905 pThis->IMount.pfnIsMounted = drvHostBaseIsMounted;
1906 pThis->IMount.pfnLock = drvHostBaseLock;
1907 pThis->IMount.pfnUnlock = drvHostBaseUnlock;
1908 pThis->IMount.pfnIsLocked = drvHostBaseIsLocked;
1909
1910 /*
1911 * Get the IBlockPort & IMountNotify interfaces of the above driver/device.
1912 */
1913 pThis->pDrvBlockPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIBLOCKPORT);
1914 if (!pThis->pDrvBlockPort)
1915 {
1916 AssertMsgFailed(("Configuration error: No block port interface above!\n"));
1917 return VERR_PDM_MISSING_INTERFACE_ABOVE;
1918 }
1919 pThis->pDrvMountNotify = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIMOUNTNOTIFY);
1920
1921 /*
1922 * Query configuration.
1923 */
1924 /* Device */
1925 int rc = CFGMR3QueryStringAlloc(pCfg, "Path", &pThis->pszDevice);
1926 if (RT_FAILURE(rc))
1927 {
1928 AssertMsgFailed(("Configuration error: query for \"Path\" string returned %Rra.\n", rc));
1929 return rc;
1930 }
1931
1932 /* Mountable */
1933 uint32_t u32;
1934 rc = CFGMR3QueryU32(pCfg, "Interval", &u32);
1935 if (RT_SUCCESS(rc))
1936 pThis->cMilliesPoller = u32;
1937 else if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1938 pThis->cMilliesPoller = 1000;
1939 else if (RT_FAILURE(rc))
1940 {
1941 AssertMsgFailed(("Configuration error: Query \"Mountable\" resulted in %Rrc.\n", rc));
1942 return rc;
1943 }
1944
1945 /* ReadOnly */
1946 rc = CFGMR3QueryBool(pCfg, "ReadOnly", &pThis->fReadOnlyConfig);
1947 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1948 pThis->fReadOnlyConfig = enmType == PDMBLOCKTYPE_DVD || enmType == PDMBLOCKTYPE_CDROM ? true : false;
1949 else if (RT_FAILURE(rc))
1950 {
1951 AssertMsgFailed(("Configuration error: Query \"ReadOnly\" resulted in %Rrc.\n", rc));
1952 return rc;
1953 }
1954
1955 /* Locked */
1956 rc = CFGMR3QueryBool(pCfg, "Locked", &pThis->fLocked);
1957 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1958 pThis->fLocked = false;
1959 else if (RT_FAILURE(rc))
1960 {
1961 AssertMsgFailed(("Configuration error: Query \"Locked\" resulted in %Rrc.\n", rc));
1962 return rc;
1963 }
1964
1965 /* BIOS visible */
1966 rc = CFGMR3QueryBool(pCfg, "BIOSVisible", &pThis->fBiosVisible);
1967 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1968 pThis->fBiosVisible = true;
1969 else if (RT_FAILURE(rc))
1970 {
1971 AssertMsgFailed(("Configuration error: Query \"BIOSVisible\" resulted in %Rrc.\n", rc));
1972 return rc;
1973 }
1974
1975 /* Uuid */
1976 char *psz;
1977 rc = CFGMR3QueryStringAlloc(pCfg, "Uuid", &psz);
1978 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1979 RTUuidClear(&pThis->Uuid);
1980 else if (RT_SUCCESS(rc))
1981 {
1982 rc = RTUuidFromStr(&pThis->Uuid, psz);
1983 if (RT_FAILURE(rc))
1984 {
1985 AssertMsgFailed(("Configuration error: Uuid from string failed on \"%s\", rc=%Rrc.\n", psz, rc));
1986 MMR3HeapFree(psz);
1987 return rc;
1988 }
1989 MMR3HeapFree(psz);
1990 }
1991 else
1992 {
1993 AssertMsgFailed(("Configuration error: Failed to obtain the uuid, rc=%Rrc.\n", rc));
1994 return rc;
1995 }
1996
1997 /* Define whether attach failure is an error (default) or not. */
1998 bool fAttachFailError;
1999 rc = CFGMR3QueryBool(pCfg, "AttachFailError", &fAttachFailError);
2000 if (RT_FAILURE(rc))
2001 fAttachFailError = true;
2002 pThis->fAttachFailError = fAttachFailError;
2003
2004 /* name to open & watch for */
2005#ifdef RT_OS_WINDOWS
2006 int iBit = RT_C_TO_UPPER(pThis->pszDevice[0]) - 'A';
2007 if ( iBit > 'Z' - 'A'
2008 || pThis->pszDevice[1] != ':'
2009 || pThis->pszDevice[2])
2010 {
2011 AssertMsgFailed(("Configuration error: Invalid drive specification: '%s'\n", pThis->pszDevice));
2012 return VERR_INVALID_PARAMETER;
2013 }
2014 pThis->fUnitMask = 1 << iBit;
2015 RTStrAPrintf(&pThis->pszDeviceOpen, "\\\\.\\%s", pThis->pszDevice);
2016
2017#elif defined(RT_OS_SOLARIS)
2018 char *pszBlockDevName = getfullblkname(pThis->pszDevice);
2019 if (!pszBlockDevName)
2020 return VERR_NO_MEMORY;
2021 pThis->pszDeviceOpen = RTStrDup(pszBlockDevName); /* for RTStrFree() */
2022 free(pszBlockDevName);
2023 pThis->pszRawDeviceOpen = RTStrDup(pThis->pszDevice);
2024
2025#else
2026 pThis->pszDeviceOpen = RTStrDup(pThis->pszDevice);
2027#endif
2028
2029 if (!pThis->pszDeviceOpen)
2030 return VERR_NO_MEMORY;
2031
2032 return VINF_SUCCESS;
2033}
2034
2035
2036/**
2037 * Do the 2nd part of the init after the derived driver has overridden the defaults.
2038 *
2039 * On failure call DRVHostBaseDestruct().
2040 *
2041 * @returns VBox status code.
2042 * @param pThis Pointer to the instance data.
2043 */
2044int DRVHostBaseInitFinish(PDRVHOSTBASE pThis)
2045{
2046 int src = VINF_SUCCESS;
2047 PPDMDRVINS pDrvIns = pThis->pDrvIns;
2048
2049 /* log config summary */
2050 Log(("%s-%d: pszDevice='%s' (%s) cMilliesPoller=%d fReadOnlyConfig=%d fLocked=%d fBIOSVisible=%d Uuid=%RTuuid\n",
2051 pDrvIns->pReg->szName, pDrvIns->iInstance, pThis->pszDevice, pThis->pszDeviceOpen, pThis->cMilliesPoller,
2052 pThis->fReadOnlyConfig, pThis->fLocked, pThis->fBiosVisible, &pThis->Uuid));
2053
2054 /*
2055 * Check that there are no drivers below us.
2056 */
2057 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
2058 ("Configuration error: Not possible to attach anything to this driver!\n"),
2059 VERR_PDM_DRVINS_NO_ATTACH);
2060
2061 /*
2062 * Register saved state.
2063 */
2064 int rc = PDMDrvHlpSSMRegisterLoadDone(pDrvIns, drvHostBaseLoadDone);
2065 if (RT_FAILURE(rc))
2066 return rc;
2067
2068 /*
2069 * Verify type.
2070 */
2071#ifdef RT_OS_WINDOWS
2072 UINT uDriveType = GetDriveType(pThis->pszDevice);
2073 switch (pThis->enmType)
2074 {
2075 case PDMBLOCKTYPE_FLOPPY_360:
2076 case PDMBLOCKTYPE_FLOPPY_720:
2077 case PDMBLOCKTYPE_FLOPPY_1_20:
2078 case PDMBLOCKTYPE_FLOPPY_1_44:
2079 case PDMBLOCKTYPE_FLOPPY_2_88:
2080 case PDMBLOCKTYPE_FLOPPY_FAKE_15_6:
2081 case PDMBLOCKTYPE_FLOPPY_FAKE_63_5:
2082 if (uDriveType != DRIVE_REMOVABLE)
2083 {
2084 AssertMsgFailed(("Configuration error: '%s' is not a floppy (type=%d)\n",
2085 pThis->pszDevice, uDriveType));
2086 return VERR_INVALID_PARAMETER;
2087 }
2088 break;
2089 case PDMBLOCKTYPE_CDROM:
2090 case PDMBLOCKTYPE_DVD:
2091 if (uDriveType != DRIVE_CDROM)
2092 {
2093 AssertMsgFailed(("Configuration error: '%s' is not a cdrom (type=%d)\n",
2094 pThis->pszDevice, uDriveType));
2095 return VERR_INVALID_PARAMETER;
2096 }
2097 break;
2098 case PDMBLOCKTYPE_HARD_DISK:
2099 default:
2100 AssertMsgFailed(("enmType=%d\n", pThis->enmType));
2101 return VERR_INVALID_PARAMETER;
2102 }
2103#endif
2104
2105 /*
2106 * Open the device.
2107 */
2108#if defined(RT_OS_DARWIN)
2109 rc = drvHostBaseOpen(pThis, NULL, pThis->fReadOnlyConfig);
2110#else
2111 rc = drvHostBaseReopen(pThis);
2112#endif
2113 if (RT_FAILURE(rc))
2114 {
2115 char *pszDevice = pThis->pszDevice;
2116#ifndef RT_OS_DARWIN
2117 char szPathReal[256];
2118 if ( RTPathExists(pszDevice)
2119 && RT_SUCCESS(RTPathReal(pszDevice, szPathReal, sizeof(szPathReal))))
2120 pszDevice = szPathReal;
2121 pThis->hFileDevice = NIL_RTFILE;
2122#endif
2123#ifdef RT_OS_SOLARIS
2124 pThis->hFileRawDevice = NIL_RTFILE;
2125#endif
2126
2127 /*
2128 * Disable CD/DVD passthrough in case it was enabled. Would cause
2129 * weird failures later when the guest issues commands. These would
2130 * all fail because of the invalid file handle. So use the normal
2131 * virtual CD/DVD code, which deals more gracefully with unavailable
2132 * "media" - actually a complete drive in this case.
2133 */
2134 pThis->IBlock.pfnSendCmd = NULL;
2135 AssertMsgFailed(("Could not open host device %s, rc=%Rrc\n", pszDevice, rc));
2136 switch (rc)
2137 {
2138 case VERR_ACCESS_DENIED:
2139 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
2140#ifdef RT_OS_LINUX
2141 N_("Cannot open host device '%s' for %s access. Check the permissions "
2142 "of that device ('/bin/ls -l %s'): Most probably you need to be member "
2143 "of the device group. Make sure that you logout/login after changing "
2144 "the group settings of the current user"),
2145#else
2146 N_("Cannot open host device '%s' for %s access. Check the permissions "
2147 "of that device"),
2148#endif
2149 pszDevice, pThis->fReadOnlyConfig ? "readonly" : "read/write",
2150 pszDevice);
2151 default:
2152 {
2153 if (pThis->fAttachFailError)
2154 return rc;
2155 int erc = PDMDrvHlpVMSetRuntimeError(pDrvIns, 0 /*fFlags*/,
2156 "DrvHost_MOUNTFAIL",
2157 N_("Cannot attach to host device '%s'"), pszDevice);
2158 AssertRC(erc);
2159 src = rc;
2160 }
2161 }
2162 }
2163#ifdef RT_OS_WINDOWS
2164 if (RT_SUCCESS(src))
2165 DRVHostBaseMediaPresent(pThis);
2166#endif
2167
2168 /*
2169 * Lock the drive if that's required by the configuration.
2170 */
2171 if (pThis->fLocked)
2172 {
2173 if (pThis->pfnDoLock)
2174 rc = pThis->pfnDoLock(pThis, true);
2175 if (RT_FAILURE(rc))
2176 {
2177 AssertMsgFailed(("Failed to lock the dvd drive. rc=%Rrc\n", rc));
2178 return rc;
2179 }
2180 }
2181
2182#ifndef RT_OS_WINDOWS
2183 if (RT_SUCCESS(src))
2184 {
2185 /*
2186 * Create the event semaphore which the poller thread will wait on.
2187 */
2188 rc = RTSemEventCreate(&pThis->EventPoller);
2189 if (RT_FAILURE(rc))
2190 return rc;
2191 }
2192#endif
2193
2194 /*
2195 * Initialize the critical section used for serializing the access to the media.
2196 */
2197 rc = RTCritSectInit(&pThis->CritSect);
2198 if (RT_FAILURE(rc))
2199 return rc;
2200
2201 if (RT_SUCCESS(src))
2202 {
2203 /*
2204 * Start the thread which will poll for the media.
2205 */
2206 rc = RTThreadCreate(&pThis->ThreadPoller, drvHostBaseMediaThread, pThis, 0,
2207 RTTHREADTYPE_INFREQUENT_POLLER, RTTHREADFLAGS_WAITABLE, "DVDMEDIA");
2208 if (RT_FAILURE(rc))
2209 {
2210 AssertMsgFailed(("Failed to create poller thread. rc=%Rrc\n", rc));
2211 return rc;
2212 }
2213
2214 /*
2215 * Wait for the thread to start up (!w32:) and do one detection loop.
2216 */
2217 rc = RTThreadUserWait(pThis->ThreadPoller, 10000);
2218 AssertRC(rc);
2219#ifdef RT_OS_WINDOWS
2220 if (!pThis->hwndDeviceChange)
2221 return VERR_GENERAL_FAILURE;
2222#endif
2223 }
2224
2225 if (RT_FAILURE(src))
2226 return src;
2227 return rc;
2228}
2229
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