VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvHostBase-darwin.cpp@ 64831

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

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

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 29.0 KB
Line 
1/* $Id: DrvHostBase-darwin.cpp 64316 2016-10-19 11:59:42Z vboxsync $ */
2/** @file
3 * DrvHostBase - Host base drive access driver, OS X specifics.
4 */
5
6/*
7 * Copyright (C) 2006-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17#define LOG_GROUP LOG_GROUP_DRV_HOST_BASE
18#include <mach/mach.h>
19#include <Carbon/Carbon.h>
20#include <IOKit/IOKitLib.h>
21#include <IOKit/storage/IOStorageDeviceCharacteristics.h>
22#include <IOKit/scsi/SCSITaskLib.h>
23#include <IOKit/scsi/SCSICommandOperationCodes.h>
24#include <IOKit/IOBSD.h>
25#include <DiskArbitration/DiskArbitration.h>
26#include <mach/mach_error.h>
27#include <VBox/scsi.h>
28
29/**
30 * Host backend specific data.
31 */
32typedef struct DRVHOSTBASEOS
33{
34 /** The master port. */
35 mach_port_t MasterPort;
36 /** The MMC-2 Device Interface. (This is only used to get the scsi task interface.) */
37 MMCDeviceInterface **ppMMCDI;
38 /** The SCSI Task Device Interface. */
39 SCSITaskDeviceInterface **ppScsiTaskDI;
40 /** The block size. Set when querying the media size. */
41 uint32_t cbBlock;
42 /** The disk arbitration session reference. NULL if we didn't have to claim & unmount the device. */
43 DASessionRef pDASession;
44 /** The disk arbitration disk reference. NULL if we didn't have to claim & unmount the device. */
45 DADiskRef pDADisk;
46 /** The number of errors that could go into the release log. (flood gate) */
47 uint32_t cLogRelErrors;
48} DRVHOSTBASEOS;
49/** Pointer to the host backend specific data. */
50typedef DRVHOSTBASEOS *PDRVHOSBASEOS;
51AssertCompile(sizeof(DRVHOSTBASEOS) <= 64);
52
53#define DRVHOSTBASE_OS_INT_DECLARED
54#include "DrvHostBase.h"
55
56/** The runloop input source name for the disk arbitration events. */
57# 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. */
58
59/**
60 * Gets the BSD Name (/dev/disc[0-9]+) for the service.
61 *
62 * This is done by recursing down the I/O registry until we hit upon an entry
63 * with a BSD Name. Usually we find it two levels down. (Further down under
64 * the IOCDPartitionScheme, the volume (slices) BSD Name is found. We don't
65 * seem to have to go this far fortunately.)
66 *
67 * @return VINF_SUCCESS if found, VERR_FILE_NOT_FOUND otherwise.
68 * @param Entry The current I/O registry entry reference.
69 * @param pszName Where to store the name. 128 bytes.
70 * @param cRecursions Number of recursions. This is used as an precaution
71 * just to limit the depth and avoid blowing the stack
72 * should we hit a bug or something.
73 */
74static int drvHostBaseGetBSDName(io_registry_entry_t Entry, char *pszName, unsigned cRecursions)
75{
76 int rc = VERR_FILE_NOT_FOUND;
77 io_iterator_t Children = 0;
78 kern_return_t krc = IORegistryEntryGetChildIterator(Entry, kIOServicePlane, &Children);
79 if (krc == KERN_SUCCESS)
80 {
81 io_object_t Child;
82 while ( rc == VERR_FILE_NOT_FOUND
83 && (Child = IOIteratorNext(Children)) != 0)
84 {
85 CFStringRef BSDNameStrRef = (CFStringRef)IORegistryEntryCreateCFProperty(Child, CFSTR(kIOBSDNameKey), kCFAllocatorDefault, 0);
86 if (BSDNameStrRef)
87 {
88 if (CFStringGetCString(BSDNameStrRef, pszName, 128, kCFStringEncodingUTF8))
89 rc = VINF_SUCCESS;
90 else
91 AssertFailed();
92 CFRelease(BSDNameStrRef);
93 }
94 if (rc == VERR_FILE_NOT_FOUND && cRecursions < 10)
95 rc = drvHostBaseGetBSDName(Child, pszName, cRecursions + 1);
96 IOObjectRelease(Child);
97 }
98 IOObjectRelease(Children);
99 }
100 return rc;
101}
102
103
104/**
105 * Callback notifying us that the async DADiskClaim()/DADiskUnmount call has completed.
106 *
107 * @param DiskRef The disk that was attempted claimed / unmounted.
108 * @param DissenterRef NULL on success, contains details on failure.
109 * @param pvContext Pointer to the return code variable.
110 */
111static void drvHostBaseDADoneCallback(DADiskRef DiskRef, DADissenterRef DissenterRef, void *pvContext)
112{
113 RT_NOREF(DiskRef);
114 int *prc = (int *)pvContext;
115 if (!DissenterRef)
116 *prc = 0;
117 else
118 *prc = DADissenterGetStatus(DissenterRef) ? DADissenterGetStatus(DissenterRef) : -1;
119 CFRunLoopStop(CFRunLoopGetCurrent());
120}
121
122
123/**
124 * Obtain exclusive access to the DVD device, umount it if necessary.
125 *
126 * @return VBox status code.
127 * @param pThis The driver instance.
128 * @param DVDService The DVD service object.
129 */
130static int drvHostBaseObtainExclusiveAccess(PDRVHOSTBASE pThis, io_object_t DVDService)
131{
132 PPDMDRVINS pDrvIns = pThis->pDrvIns; NOREF(pDrvIns);
133
134 for (unsigned iTry = 0;; iTry++)
135 {
136 IOReturn irc = (*pThis->Os.ppScsiTaskDI)->ObtainExclusiveAccess(pThis->Os.ppScsiTaskDI);
137 if (irc == kIOReturnSuccess)
138 {
139 /*
140 * This is a bit weird, but if we unmounted the DVD drive we also need to
141 * unlock it afterwards or the guest won't be able to eject it later on.
142 */
143 if (pThis->Os.pDADisk)
144 {
145 uint8_t abCmd[16] =
146 {
147 SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL, 0, 0, 0, false, 0,
148 0,0,0,0,0,0,0,0,0,0
149 };
150 drvHostBaseScsiCmdOs(pThis, abCmd, 6, PDMMEDIATXDIR_NONE, NULL, NULL, NULL, 0, 0);
151 }
152 return VINF_SUCCESS;
153 }
154 if (irc == kIOReturnExclusiveAccess)
155 return VERR_SHARING_VIOLATION; /* already used exclusivly. */
156 if (irc != kIOReturnBusy)
157 return VERR_GENERAL_FAILURE; /* not mounted */
158
159 /*
160 * Attempt to the unmount all volumes of the device.
161 * It seems we can can do this all in one go without having to enumerate the
162 * volumes (sessions) and deal with them one by one. This is very fortuitous
163 * as the disk arbitration API is a bit cumbersome to deal with.
164 */
165 if (iTry > 2)
166 return VERR_DRIVE_LOCKED;
167 char szName[128];
168 int rc = drvHostBaseGetBSDName(DVDService, &szName[0], 0);
169 if (RT_SUCCESS(rc))
170 {
171 pThis->Os.pDASession = DASessionCreate(kCFAllocatorDefault);
172 if (pThis->Os.pDASession)
173 {
174 DASessionScheduleWithRunLoop(pThis->Os.pDASession, CFRunLoopGetCurrent(), MY_RUN_LOOP_MODE);
175 pThis->Os.pDADisk = DADiskCreateFromBSDName(kCFAllocatorDefault, pThis->Os.pDASession, szName);
176 if (pThis->Os.pDADisk)
177 {
178 /*
179 * Try claim the device.
180 */
181 Log(("%s-%d: calling DADiskClaim on '%s'.\n", pDrvIns->pReg->szName, pDrvIns->iInstance, szName));
182 int rcDA = -2;
183 DADiskClaim(pThis->Os.pDADisk, kDADiskClaimOptionDefault, NULL, NULL, drvHostBaseDADoneCallback, &rcDA);
184 SInt32 rc32 = CFRunLoopRunInMode(MY_RUN_LOOP_MODE, 120.0, FALSE);
185 AssertMsg(rc32 == kCFRunLoopRunStopped, ("rc32=%RI32 (%RX32)\n", rc32, rc32));
186 if ( rc32 == kCFRunLoopRunStopped
187 && !rcDA)
188 {
189 /*
190 * Try unmount the device.
191 */
192 Log(("%s-%d: calling DADiskUnmount on '%s'.\n", pDrvIns->pReg->szName, pDrvIns->iInstance, szName));
193 rcDA = -2;
194 DADiskUnmount(pThis->Os.pDADisk, kDADiskUnmountOptionWhole, drvHostBaseDADoneCallback, &rcDA);
195 rc32 = CFRunLoopRunInMode(MY_RUN_LOOP_MODE, 120.0, FALSE);
196 AssertMsg(rc32 == kCFRunLoopRunStopped, ("rc32=%RI32 (%RX32)\n", rc32, rc32));
197 if ( rc32 == kCFRunLoopRunStopped
198 && !rcDA)
199 {
200 iTry = 99;
201 DASessionUnscheduleFromRunLoop(pThis->Os.pDASession, CFRunLoopGetCurrent(), MY_RUN_LOOP_MODE);
202 Log(("%s-%d: unmount succeed - retrying.\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
203 continue;
204 }
205 Log(("%s-%d: umount => rc32=%d & rcDA=%#x\n", pDrvIns->pReg->szName, pDrvIns->iInstance, rc32, rcDA));
206
207 /* failed - cleanup */
208 DADiskUnclaim(pThis->Os.pDADisk);
209 }
210 else
211 Log(("%s-%d: claim => rc32=%d & rcDA=%#x\n", pDrvIns->pReg->szName, pDrvIns->iInstance, rc32, rcDA));
212
213 CFRelease(pThis->Os.pDADisk);
214 pThis->Os.pDADisk = NULL;
215 }
216 else
217 Log(("%s-%d: failed to open disk '%s'!\n", pDrvIns->pReg->szName, pDrvIns->iInstance, szName));
218
219 DASessionUnscheduleFromRunLoop(pThis->Os.pDASession, CFRunLoopGetCurrent(), MY_RUN_LOOP_MODE);
220 CFRelease(pThis->Os.pDASession);
221 pThis->Os.pDASession = NULL;
222 }
223 else
224 Log(("%s-%d: failed to create DA session!\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
225 }
226 RTThreadSleep(10);
227 }
228}
229
230DECLHIDDEN(int) drvHostBaseScsiCmdOs(PDRVHOSTBASE pThis, const uint8_t *pbCmd, size_t cbCmd, PDMMEDIATXDIR enmTxDir,
231 void *pvBuf, uint32_t *pcbBuf, uint8_t *pbSense, size_t cbSense, uint32_t cTimeoutMillies)
232{
233 /*
234 * Minimal input validation.
235 */
236 Assert(enmTxDir == PDMMEDIATXDIR_NONE || enmTxDir == PDMMEDIATXDIR_FROM_DEVICE || enmTxDir == PDMMEDIATXDIR_TO_DEVICE);
237 Assert(!pvBuf || pcbBuf);
238 Assert(pvBuf || enmTxDir == PDMMEDIATXDIR_NONE);
239 Assert(pbSense || !cbSense);
240 AssertPtr(pbCmd);
241 Assert(cbCmd <= 16 && cbCmd >= 1);
242 const uint32_t cbBuf = pcbBuf ? *pcbBuf : 0;
243 if (pcbBuf)
244 *pcbBuf = 0;
245
246 Assert(pThis->Os.ppScsiTaskDI);
247
248 int rc = VERR_GENERAL_FAILURE;
249 SCSITaskInterface **ppScsiTaskI = (*pThis->Os.ppScsiTaskDI)->CreateSCSITask(pThis->Os.ppScsiTaskDI);
250 if (!ppScsiTaskI)
251 return VERR_NO_MEMORY;
252 do
253 {
254 /* Setup the scsi command. */
255 SCSICommandDescriptorBlock cdb = { 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 };
256 memcpy(&cdb[0], pbCmd, cbCmd);
257 IOReturn irc = (*ppScsiTaskI)->SetCommandDescriptorBlock(ppScsiTaskI, cdb, cbCmd);
258 AssertBreak(irc == kIOReturnSuccess);
259
260 /* Setup the buffer. */
261 if (enmTxDir == PDMMEDIATXDIR_NONE)
262 irc = (*ppScsiTaskI)->SetScatterGatherEntries(ppScsiTaskI, NULL, 0, 0, kSCSIDataTransfer_NoDataTransfer);
263 else
264 {
265 IOVirtualRange Range = { (IOVirtualAddress)pvBuf, cbBuf };
266 irc = (*ppScsiTaskI)->SetScatterGatherEntries(ppScsiTaskI, &Range, 1, cbBuf,
267 enmTxDir == PDMMEDIATXDIR_FROM_DEVICE
268 ? kSCSIDataTransfer_FromTargetToInitiator
269 : kSCSIDataTransfer_FromInitiatorToTarget);
270 }
271 AssertBreak(irc == kIOReturnSuccess);
272
273 /* Set the timeout. */
274 irc = (*ppScsiTaskI)->SetTimeoutDuration(ppScsiTaskI, cTimeoutMillies ? cTimeoutMillies : 30000 /*ms*/);
275 AssertBreak(irc == kIOReturnSuccess);
276
277 /* Execute the command and get the response. */
278 SCSI_Sense_Data SenseData = { 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 };
279 SCSIServiceResponse ServiceResponse = kSCSIServiceResponse_Request_In_Process;
280 SCSITaskStatus TaskStatus = kSCSITaskStatus_GOOD;
281 UInt64 cbReturned = 0;
282 irc = (*ppScsiTaskI)->ExecuteTaskSync(ppScsiTaskI, &SenseData, &TaskStatus, &cbReturned);
283 AssertBreak(irc == kIOReturnSuccess);
284 if (pcbBuf)
285 *pcbBuf = (int32_t)cbReturned;
286
287 irc = (*ppScsiTaskI)->GetSCSIServiceResponse(ppScsiTaskI, &ServiceResponse);
288 AssertBreak(irc == kIOReturnSuccess);
289 AssertBreak(ServiceResponse == kSCSIServiceResponse_TASK_COMPLETE);
290
291 if (TaskStatus == kSCSITaskStatus_GOOD)
292 rc = VINF_SUCCESS;
293 else if ( TaskStatus == kSCSITaskStatus_CHECK_CONDITION
294 && pbSense)
295 {
296 memset(pbSense, 0, cbSense); /* lazy */
297 memcpy(pbSense, &SenseData, RT_MIN(sizeof(SenseData), cbSense));
298 rc = VERR_UNRESOLVED_ERROR;
299 }
300 /** @todo convert sense codes when caller doesn't wish to do this himself. */
301 /*else if ( TaskStatus == kSCSITaskStatus_CHECK_CONDITION
302 && SenseData.ADDITIONAL_SENSE_CODE == 0x3A)
303 rc = VERR_MEDIA_NOT_PRESENT; */
304 else
305 {
306 rc = enmTxDir == PDMMEDIATXDIR_NONE
307 ? VERR_DEV_IO_ERROR
308 : enmTxDir == PDMMEDIATXDIR_FROM_DEVICE
309 ? VERR_READ_ERROR
310 : VERR_WRITE_ERROR;
311 if (pThis->Os.cLogRelErrors++ < 10)
312 LogRel(("DVD scsi error: cmd={%.*Rhxs} TaskStatus=%#x key=%#x ASC=%#x ASCQ=%#x (%Rrc)\n",
313 cbCmd, pbCmd, TaskStatus, SenseData.SENSE_KEY, SenseData.ADDITIONAL_SENSE_CODE,
314 SenseData.ADDITIONAL_SENSE_CODE_QUALIFIER, rc));
315 }
316 } while (0);
317
318 (*ppScsiTaskI)->Release(ppScsiTaskI);
319
320 return rc;
321}
322
323
324DECLHIDDEN(int) drvHostBaseGetMediaSizeOs(PDRVHOSTBASE pThis, uint64_t *pcb)
325{
326 /*
327 * Try a READ_CAPACITY command...
328 */
329 struct
330 {
331 uint32_t cBlocks;
332 uint32_t cbBlock;
333 } Buf = {0, 0};
334 uint32_t cbBuf = sizeof(Buf);
335 uint8_t abCmd[16] =
336 {
337 SCSI_READ_CAPACITY, 0, 0, 0, 0, 0, 0,
338 0,0,0,0,0,0,0,0,0
339 };
340 int rc = drvHostBaseScsiCmdOs(pThis, abCmd, 6, PDMMEDIATXDIR_FROM_DEVICE, &Buf, &cbBuf, NULL, 0, 0);
341 if (RT_SUCCESS(rc))
342 {
343 Assert(cbBuf == sizeof(Buf));
344 Buf.cBlocks = RT_BE2H_U32(Buf.cBlocks);
345 Buf.cbBlock = RT_BE2H_U32(Buf.cbBlock);
346 //if (Buf.cbBlock > 2048) /* everyone else is doing this... check if it needed/right.*/
347 // Buf.cbBlock = 2048;
348 pThis->Os.cbBlock = Buf.cbBlock;
349
350 *pcb = (uint64_t)Buf.cBlocks * Buf.cbBlock;
351 }
352 return rc;
353}
354
355
356DECLHIDDEN(int) drvHostBaseReadOs(PDRVHOSTBASE pThis, uint64_t off, void *pvBuf, size_t cbRead)
357{
358 int rc = VINF_SUCCESS;
359
360 if ( pThis->Os.ppScsiTaskDI
361 && pThis->Os.cbBlock)
362 {
363 /*
364 * Issue a READ(12) request.
365 */
366 do
367 {
368 const uint32_t LBA = off / pThis->Os.cbBlock;
369 AssertReturn(!(off % pThis->Os.cbBlock), VERR_INVALID_PARAMETER);
370 uint32_t cbRead32 = cbRead > SCSI_MAX_BUFFER_SIZE
371 ? SCSI_MAX_BUFFER_SIZE
372 : (uint32_t)cbRead;
373 const uint32_t cBlocks = cbRead32 / pThis->Os.cbBlock;
374 AssertReturn(!(cbRead % pThis->Os.cbBlock), VERR_INVALID_PARAMETER);
375 uint8_t abCmd[16] =
376 {
377 SCSI_READ_12, 0,
378 RT_BYTE4(LBA), RT_BYTE3(LBA), RT_BYTE2(LBA), RT_BYTE1(LBA),
379 RT_BYTE4(cBlocks), RT_BYTE3(cBlocks), RT_BYTE2(cBlocks), RT_BYTE1(cBlocks),
380 0, 0, 0, 0, 0
381 };
382 rc = drvHostBaseScsiCmdOs(pThis, abCmd, 12, PDMMEDIATXDIR_FROM_DEVICE, pvBuf, &cbRead32, NULL, 0, 0);
383
384 off += cbRead32;
385 cbRead -= cbRead32;
386 pvBuf = (uint8_t *)pvBuf + cbRead32;
387 } while ((cbRead > 0) && RT_SUCCESS(rc));
388 }
389 else
390 rc = VERR_MEDIA_NOT_PRESENT;
391
392 return rc;
393}
394
395
396DECLHIDDEN(int) drvHostBaseWriteOs(PDRVHOSTBASE pThis, uint64_t off, const void *pvBuf, size_t cbWrite)
397{
398 RT_NOREF4(pThis, off, pvBuf, cbWrite);
399 return VERR_WRITE_PROTECT;
400}
401
402
403DECLHIDDEN(int) drvHostBaseFlushOs(PDRVHOSTBASE pThis)
404{
405 RT_NOREF1(pThis);
406 return VINF_SUCCESS;
407}
408
409
410DECLHIDDEN(int) drvHostBaseDoLockOs(PDRVHOSTBASE pThis, bool fLock)
411{
412 uint8_t abCmd[16] =
413 {
414 SCSI_PREVENT_ALLOW_MEDIUM_REMOVAL, 0, 0, 0, fLock, 0,
415 0,0,0,0,0,0,0,0,0,0
416 };
417 return drvHostBaseScsiCmdOs(pThis, abCmd, 6, PDMMEDIATXDIR_NONE, NULL, NULL, NULL, 0, 0);
418}
419
420
421DECLHIDDEN(int) drvHostBaseEjectOs(PDRVHOSTBASE pThis)
422{
423 uint8_t abCmd[16] =
424 {
425 SCSI_START_STOP_UNIT, 0, 0, 0, 2 /*eject+stop*/, 0,
426 0,0,0,0,0,0,0,0,0,0
427 };
428 return drvHostBaseScsiCmdOs(pThis, abCmd, 6, PDMMEDIATXDIR_NONE, NULL, NULL, NULL, 0, 0);
429}
430
431
432DECLHIDDEN(int) drvHostBaseQueryMediaStatusOs(PDRVHOSTBASE pThis, bool *pfMediaChanged, bool *pfMediaPresent)
433{
434 AssertReturn(pThis->Os.ppScsiTaskDI, VERR_INTERNAL_ERROR);
435
436 /*
437 * Issue a TEST UNIT READY request.
438 */
439 *pfMediaChanged = false;
440 *pfMediaPresent = false;
441 uint8_t abCmd[16] = { SCSI_TEST_UNIT_READY, 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0 };
442 uint8_t abSense[32];
443 int rc = drvHostBaseScsiCmdOs(pThis, abCmd, 6, PDMMEDIATXDIR_NONE, NULL, NULL, abSense, sizeof(abSense), 0);
444 if (RT_SUCCESS(rc))
445 *pfMediaPresent = true;
446 else if ( rc == VERR_UNRESOLVED_ERROR
447 && abSense[2] == 6 /* unit attention */
448 && ( (abSense[12] == 0x29 && abSense[13] < 5 /* reset */)
449 || (abSense[12] == 0x2a && abSense[13] == 0 /* parameters changed */) //???
450 || (abSense[12] == 0x3f && abSense[13] == 0 /* target operating conditions have changed */) //???
451 || (abSense[12] == 0x3f && abSense[13] == 2 /* changed operating definition */) //???
452 || (abSense[12] == 0x3f && abSense[13] == 3 /* inquiry parameters changed */)
453 || (abSense[12] == 0x3f && abSense[13] == 5 /* device identifier changed */)
454 )
455 )
456 {
457 *pfMediaPresent = false;
458 *pfMediaChanged = true;
459 rc = VINF_SUCCESS;
460 /** @todo check this media change stuff on Darwin. */
461 }
462
463 return rc;
464}
465
466
467DECLHIDDEN(void) drvHostBaseInitOs(PDRVHOSTBASE pThis)
468{
469 pThis->Os.MasterPort = IO_OBJECT_NULL;
470 pThis->Os.ppMMCDI = NULL;
471 pThis->Os.ppScsiTaskDI = NULL;
472 pThis->Os.cbBlock = 0;
473 pThis->Os.pDADisk = NULL;
474 pThis->Os.pDASession = NULL;
475}
476
477
478DECLHIDDEN(int) drvHostBaseOpenOs(PDRVHOSTBASE pThis, bool fReadOnly)
479{
480 RT_NOREF(fReadOnly);
481
482 /* Darwin is kind of special... */
483 Assert(!pThis->Os.cbBlock);
484 Assert(pThis->Os.MasterPort == IO_OBJECT_NULL);
485 Assert(!pThis->Os.ppMMCDI);
486 Assert(!pThis->Os.ppScsiTaskDI);
487
488 /*
489 * Open the master port on the first invocation.
490 */
491 kern_return_t krc = IOMasterPort(MACH_PORT_NULL, &pThis->Os.MasterPort);
492 AssertReturn(krc == KERN_SUCCESS, VERR_GENERAL_FAILURE);
493
494 /*
495 * Create a matching dictionary for searching for CD, DVD and BlueRay services in the IOKit.
496 *
497 * The idea is to find all the devices which are of class IOCDBlockStorageDevice.
498 * CD devices are represented by IOCDBlockStorageDevice class itself, while DVD and BlueRay ones
499 * have it as a parent class.
500 */
501 CFMutableDictionaryRef RefMatchingDict = IOServiceMatching("IOCDBlockStorageDevice");
502 AssertReturn(RefMatchingDict, VERR_NOT_FOUND);
503
504 /*
505 * do the search and get a collection of keyboards.
506 */
507 io_iterator_t DVDServices = IO_OBJECT_NULL;
508 IOReturn irc = IOServiceGetMatchingServices(pThis->Os.MasterPort, RefMatchingDict, &DVDServices);
509 AssertMsgReturn(irc == kIOReturnSuccess, ("irc=%d\n", irc), VERR_NOT_FOUND);
510 RefMatchingDict = NULL; /* the reference is consumed by IOServiceGetMatchingServices. */
511
512 /*
513 * Enumerate the matching drives (services).
514 * (This enumeration must be identical to the one performed in Main/src-server/darwin/iokit.cpp.)
515 */
516 int rc = VERR_FILE_NOT_FOUND;
517 unsigned i = 0;
518 io_object_t DVDService;
519 while ((DVDService = IOIteratorNext(DVDServices)) != 0)
520 {
521 /*
522 * Get the properties we use to identify the DVD drive.
523 *
524 * While there is a (weird 12 byte) GUID, it isn't persistent
525 * across boots. So, we have to use a combination of the
526 * vendor name and product name properties with an optional
527 * sequence number for identification.
528 */
529 CFMutableDictionaryRef PropsRef = 0;
530 krc = IORegistryEntryCreateCFProperties(DVDService, &PropsRef, kCFAllocatorDefault, kNilOptions);
531 if (krc == KERN_SUCCESS)
532 {
533 /* Get the Device Characteristics dictionary. */
534 CFDictionaryRef DevCharRef = (CFDictionaryRef)CFDictionaryGetValue(PropsRef, CFSTR(kIOPropertyDeviceCharacteristicsKey));
535 if (DevCharRef)
536 {
537 /* The vendor name. */
538 char szVendor[128];
539 char *pszVendor = &szVendor[0];
540 CFTypeRef ValueRef = CFDictionaryGetValue(DevCharRef, CFSTR(kIOPropertyVendorNameKey));
541 if ( ValueRef
542 && CFGetTypeID(ValueRef) == CFStringGetTypeID()
543 && CFStringGetCString((CFStringRef)ValueRef, szVendor, sizeof(szVendor), kCFStringEncodingUTF8))
544 pszVendor = RTStrStrip(szVendor);
545 else
546 *pszVendor = '\0';
547
548 /* The product name. */
549 char szProduct[128];
550 char *pszProduct = &szProduct[0];
551 ValueRef = CFDictionaryGetValue(DevCharRef, CFSTR(kIOPropertyProductNameKey));
552 if ( ValueRef
553 && CFGetTypeID(ValueRef) == CFStringGetTypeID()
554 && CFStringGetCString((CFStringRef)ValueRef, szProduct, sizeof(szProduct), kCFStringEncodingUTF8))
555 pszProduct = RTStrStrip(szProduct);
556 else
557 *pszProduct = '\0';
558
559 /* Construct the two names and compare thwm with the one we're searching for. */
560 char szName1[256 + 32];
561 char szName2[256 + 32];
562 if (*pszVendor || *pszProduct)
563 {
564 if (*pszVendor && *pszProduct)
565 {
566 RTStrPrintf(szName1, sizeof(szName1), "%s %s", pszVendor, pszProduct);
567 RTStrPrintf(szName2, sizeof(szName2), "%s %s (#%u)", pszVendor, pszProduct, i);
568 }
569 else
570 {
571 strcpy(szName1, *pszVendor ? pszVendor : pszProduct);
572 RTStrPrintf(szName2, sizeof(szName2), "%s (#%u)", *pszVendor ? pszVendor : pszProduct, i);
573 }
574 }
575 else
576 {
577 RTStrPrintf(szName1, sizeof(szName1), "(#%u)", i);
578 strcpy(szName2, szName1);
579 }
580
581 if ( !strcmp(szName1, pThis->pszDevice)
582 || !strcmp(szName2, pThis->pszDevice))
583 {
584 /*
585 * Found it! Now, get the client interface and stuff.
586 * Note that we could also query kIOSCSITaskDeviceUserClientTypeID here if the
587 * MMC client plugin is missing. For now we assume this won't be necessary.
588 */
589 SInt32 Score = 0;
590 IOCFPlugInInterface **ppPlugInInterface = NULL;
591 krc = IOCreatePlugInInterfaceForService(DVDService, kIOMMCDeviceUserClientTypeID, kIOCFPlugInInterfaceID,
592 &ppPlugInInterface, &Score);
593 if (krc == KERN_SUCCESS)
594 {
595 HRESULT hrc = (*ppPlugInInterface)->QueryInterface(ppPlugInInterface,
596 CFUUIDGetUUIDBytes(kIOMMCDeviceInterfaceID),
597 (LPVOID *)&pThis->Os.ppMMCDI);
598 (*ppPlugInInterface)->Release(ppPlugInInterface);
599 ppPlugInInterface = NULL;
600 if (hrc == S_OK)
601 {
602 pThis->Os.ppScsiTaskDI = (*pThis->Os.ppMMCDI)->GetSCSITaskDeviceInterface(pThis->Os.ppMMCDI);
603 if (pThis->Os.ppScsiTaskDI)
604 rc = VINF_SUCCESS;
605 else
606 {
607 LogRel(("GetSCSITaskDeviceInterface failed on '%s'\n", pThis->pszDevice));
608 rc = VERR_NOT_SUPPORTED;
609 (*pThis->Os.ppMMCDI)->Release(pThis->Os.ppMMCDI);
610 }
611 }
612 else
613 {
614 rc = VERR_GENERAL_FAILURE;//RTErrConvertFromDarwinCOM(krc);
615 pThis->Os.ppMMCDI = NULL;
616 }
617 }
618 else /* Check for kIOSCSITaskDeviceUserClientTypeID? */
619 rc = VERR_GENERAL_FAILURE;//RTErrConvertFromDarwinKern(krc);
620
621 /* Obtain exclusive access to the device so we can send SCSI commands. */
622 if (RT_SUCCESS(rc))
623 rc = drvHostBaseObtainExclusiveAccess(pThis, DVDService);
624
625 /* Cleanup on failure. */
626 if (RT_FAILURE(rc))
627 {
628 if (pThis->Os.ppScsiTaskDI)
629 {
630 (*pThis->Os.ppScsiTaskDI)->Release(pThis->Os.ppScsiTaskDI);
631 pThis->Os.ppScsiTaskDI = NULL;
632 }
633 if (pThis->Os.ppMMCDI)
634 {
635 (*pThis->Os.ppMMCDI)->Release(pThis->Os.ppMMCDI);
636 pThis->Os.ppMMCDI = NULL;
637 }
638 }
639
640 IOObjectRelease(DVDService);
641 break;
642 }
643 }
644 CFRelease(PropsRef);
645 }
646 else
647 AssertMsgFailed(("krc=%#x\n", krc));
648
649 IOObjectRelease(DVDService);
650 i++;
651 }
652
653 IOObjectRelease(DVDServices);
654 return rc;
655
656}
657
658
659DECLHIDDEN(int) drvHostBaseMediaRefreshOs(PDRVHOSTBASE pThis)
660{
661 RT_NOREF(pThis);
662 return VINF_SUCCESS;
663}
664
665
666DECLHIDDEN(bool) drvHostBaseIsMediaPollingRequiredOs(PDRVHOSTBASE pThis)
667{
668 if (pThis->enmType == PDMMEDIATYPE_CDROM || pThis->enmType == PDMMEDIATYPE_DVD)
669 return true;
670
671 AssertMsgFailed(("Darwin supports only CD/DVD host drive access\n"));
672 return false;
673}
674
675
676DECLHIDDEN(void) drvHostBaseDestructOs(PDRVHOSTBASE pThis)
677{
678 /*
679 * Unlock the drive if we've locked it or we're in passthru mode.
680 */
681 if ( ( pThis->fLocked
682 || pThis->IMedia.pfnSendCmd)
683 && pThis->Os.ppScsiTaskDI
684 && pThis->pfnDoLock)
685 {
686 int rc = pThis->pfnDoLock(pThis, false);
687 if (RT_SUCCESS(rc))
688 pThis->fLocked = false;
689 }
690
691 /*
692 * The unclaiming doesn't seem to mean much, the DVD is actually
693 * remounted when we release exclusive access. I'm not quite sure
694 * if I should put the unclaim first or not...
695 *
696 * Anyway, that it's automatically remounted very good news for us,
697 * because that means we don't have to mess with that ourselves. Of
698 * course there is the unlikely scenario that we've succeeded in claiming
699 * and umount the DVD but somehow failed to gain exclusive scsi access...
700 */
701 if (pThis->Os.ppScsiTaskDI)
702 {
703 LogFlow(("%s-%d: releasing exclusive scsi access!\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
704 (*pThis->Os.ppScsiTaskDI)->ReleaseExclusiveAccess(pThis->Os.ppScsiTaskDI);
705 (*pThis->Os.ppScsiTaskDI)->Release(pThis->Os.ppScsiTaskDI);
706 pThis->Os.ppScsiTaskDI = NULL;
707 }
708 if (pThis->Os.pDADisk)
709 {
710 LogFlow(("%s-%d: unclaiming the disk!\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
711 DADiskUnclaim(pThis->Os.pDADisk);
712 CFRelease(pThis->Os.pDADisk);
713 pThis->Os.pDADisk = NULL;
714 }
715 if (pThis->Os.ppMMCDI)
716 {
717 LogFlow(("%s-%d: releasing the MMC object!\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
718 (*pThis->Os.ppMMCDI)->Release(pThis->Os.ppMMCDI);
719 pThis->Os.ppMMCDI = NULL;
720 }
721 if (pThis->Os.MasterPort != IO_OBJECT_NULL)
722 {
723 mach_port_deallocate(mach_task_self(), pThis->Os.MasterPort);
724 pThis->Os.MasterPort = IO_OBJECT_NULL;
725 }
726 if (pThis->Os.pDASession)
727 {
728 LogFlow(("%s-%d: releasing the DA session!\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
729 CFRelease(pThis->Os.pDASession);
730 pThis->Os.pDASession = NULL;
731 }
732}
733
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