VirtualBox

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

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

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 52.4 KB
Line 
1/* $Id: DrvHostBase.cpp 76553 2019-01-01 01:45:53Z vboxsync $ */
2/** @file
3 * DrvHostBase - Host base drive access driver.
4 */
5
6/*
7 * Copyright (C) 2006-2019 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_DRV_HOST_BASE
23
24#include <VBox/vmm/pdmdrv.h>
25#include <VBox/vmm/pdmstorageifs.h>
26#include <VBox/err.h>
27#include <iprt/assert.h>
28#include <iprt/file.h>
29#include <iprt/path.h>
30#include <iprt/string.h>
31#include <iprt/thread.h>
32#include <iprt/semaphore.h>
33#include <iprt/uuid.h>
34#include <iprt/asm.h>
35#include <iprt/critsect.h>
36#include <iprt/ctype.h>
37#include <iprt/mem.h>
38
39#include "DrvHostBase.h"
40
41
42
43
44/* -=-=-=-=- IBlock -=-=-=-=- */
45
46/** @interface_method_impl{PDMIMEDIA,pfnRead} */
47static DECLCALLBACK(int) drvHostBaseRead(PPDMIMEDIA pInterface, uint64_t off, void *pvBuf, size_t cbRead)
48{
49 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
50 LogFlow(("%s-%d: drvHostBaseRead: off=%#llx pvBuf=%p cbRead=%#x (%s)\n",
51 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, pvBuf, cbRead, pThis->pszDevice));
52 RTCritSectEnter(&pThis->CritSect);
53
54 STAM_REL_COUNTER_INC(&pThis->StatReqsSubmitted);
55 STAM_REL_COUNTER_INC(&pThis->StatReqsRead);
56
57 /*
58 * Check the state.
59 */
60 int rc;
61 if (pThis->fMediaPresent)
62 {
63 /*
64 * Seek and read.
65 */
66 rc = drvHostBaseReadOs(pThis, off, pvBuf, cbRead);
67 if (RT_SUCCESS(rc))
68 {
69 Log2(("%s-%d: drvHostBaseReadOs: off=%#llx cbRead=%#x\n"
70 "%16.*Rhxd\n",
71 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, cbRead, cbRead, pvBuf));
72 }
73 else
74 Log(("%s-%d: drvHostBaseRead: drvHostBaseReadOs(%#llx, %p, %#x) -> %Rrc ('%s')\n",
75 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance,
76 off, pvBuf, cbRead, rc, pThis->pszDevice));
77 }
78 else
79 rc = VERR_MEDIA_NOT_PRESENT;
80
81 if (RT_SUCCESS(rc))
82 {
83 STAM_REL_COUNTER_INC(&pThis->StatReqsSucceeded);
84 STAM_REL_COUNTER_ADD(&pThis->StatBytesRead, cbRead);
85 }
86 else
87 STAM_REL_COUNTER_INC(&pThis->StatReqsFailed);
88
89 RTCritSectLeave(&pThis->CritSect);
90 LogFlow(("%s-%d: drvHostBaseRead: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
91 return rc;
92}
93
94
95/** @interface_method_impl{PDMIMEDIA,pfnWrite} */
96static DECLCALLBACK(int) drvHostBaseWrite(PPDMIMEDIA pInterface, uint64_t off, const void *pvBuf, size_t cbWrite)
97{
98 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
99 LogFlow(("%s-%d: drvHostBaseWrite: off=%#llx pvBuf=%p cbWrite=%#x (%s)\n",
100 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, pvBuf, cbWrite, pThis->pszDevice));
101 Log2(("%s-%d: drvHostBaseWrite: off=%#llx cbWrite=%#x\n"
102 "%16.*Rhxd\n",
103 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, cbWrite, cbWrite, pvBuf));
104 RTCritSectEnter(&pThis->CritSect);
105
106 STAM_REL_COUNTER_INC(&pThis->StatReqsSubmitted);
107 STAM_REL_COUNTER_INC(&pThis->StatReqsWrite);
108
109 /*
110 * Check the state.
111 */
112 int rc;
113 if (!pThis->fReadOnly)
114 {
115 if (pThis->fMediaPresent)
116 {
117 /*
118 * Seek and write.
119 */
120 rc = drvHostBaseWriteOs(pThis, off, pvBuf, cbWrite);
121 if (RT_FAILURE(rc))
122 Log(("%s-%d: drvHostBaseWrite: drvHostBaseWriteOs(%#llx, %p, %#x) -> %Rrc ('%s')\n",
123 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance,
124 off, pvBuf, cbWrite, rc, pThis->pszDevice));
125 }
126 else
127 rc = VERR_MEDIA_NOT_PRESENT;
128 }
129 else
130 rc = VERR_WRITE_PROTECT;
131
132 if (RT_SUCCESS(rc))
133 {
134 STAM_REL_COUNTER_INC(&pThis->StatReqsSucceeded);
135 STAM_REL_COUNTER_ADD(&pThis->StatBytesWritten, cbWrite);
136 }
137 else
138 STAM_REL_COUNTER_INC(&pThis->StatReqsFailed);
139
140 RTCritSectLeave(&pThis->CritSect);
141 LogFlow(("%s-%d: drvHostBaseWrite: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
142 return rc;
143}
144
145
146/** @interface_method_impl{PDMIMEDIA,pfnFlush} */
147static DECLCALLBACK(int) drvHostBaseFlush(PPDMIMEDIA pInterface)
148{
149 int rc;
150 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
151 LogFlow(("%s-%d: drvHostBaseFlush: (%s)\n",
152 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDevice));
153 RTCritSectEnter(&pThis->CritSect);
154
155 STAM_REL_COUNTER_INC(&pThis->StatReqsSubmitted);
156 STAM_REL_COUNTER_INC(&pThis->StatReqsFlush);
157
158 if (pThis->fMediaPresent)
159 rc = drvHostBaseFlushOs(pThis);
160 else
161 rc = VERR_MEDIA_NOT_PRESENT;
162
163 if (RT_SUCCESS(rc))
164 STAM_REL_COUNTER_INC(&pThis->StatReqsSucceeded);
165 else
166 STAM_REL_COUNTER_INC(&pThis->StatReqsFailed);
167
168 RTCritSectLeave(&pThis->CritSect);
169 LogFlow(("%s-%d: drvHostBaseFlush: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
170 return rc;
171}
172
173
174/** @interface_method_impl{PDMIMEDIA,pfnIsReadOnly} */
175static DECLCALLBACK(bool) drvHostBaseIsReadOnly(PPDMIMEDIA pInterface)
176{
177 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
178 return pThis->fReadOnly;
179}
180
181
182/** @interface_method_impl{PDMIMEDIA,pfnIsNonRotational} */
183static DECLCALLBACK(bool) drvHostBaseIsNonRotational(PPDMIMEDIA pInterface)
184{
185 RT_NOREF1(pInterface);
186 return false;
187}
188
189
190/** @interface_method_impl{PDMIMEDIA,pfnGetSize} */
191static DECLCALLBACK(uint64_t) drvHostBaseGetSize(PPDMIMEDIA pInterface)
192{
193 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
194 RTCritSectEnter(&pThis->CritSect);
195
196 uint64_t cb = 0;
197 if (pThis->fMediaPresent)
198 cb = pThis->cbSize;
199
200 RTCritSectLeave(&pThis->CritSect);
201 LogFlow(("%s-%d: drvHostBaseGetSize: returns %llu\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, cb));
202 return cb;
203}
204
205
206/** @interface_method_impl{PDMIMEDIA,pfnGetType} */
207static DECLCALLBACK(PDMMEDIATYPE) drvHostBaseGetType(PPDMIMEDIA pInterface)
208{
209 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
210 LogFlow(("%s-%d: drvHostBaseGetType: returns %d\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->enmType));
211 return pThis->enmType;
212}
213
214
215/** @interface_method_impl{PDMIMEDIA,pfnGetUuid} */
216static DECLCALLBACK(int) drvHostBaseGetUuid(PPDMIMEDIA pInterface, PRTUUID pUuid)
217{
218 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
219
220 *pUuid = pThis->Uuid;
221
222 LogFlow(("%s-%d: drvHostBaseGetUuid: returns VINF_SUCCESS *pUuid=%RTuuid\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pUuid));
223 return VINF_SUCCESS;
224}
225
226
227/** @interface_method_impl{PDMIMEDIA,pfnBiosGetPCHSGeometry} */
228static DECLCALLBACK(int) drvHostBaseGetPCHSGeometry(PPDMIMEDIA pInterface, PPDMMEDIAGEOMETRY pPCHSGeometry)
229{
230 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
231 RTCritSectEnter(&pThis->CritSect);
232
233 int rc = VINF_SUCCESS;
234 if (pThis->fMediaPresent)
235 {
236 if ( pThis->PCHSGeometry.cCylinders > 0
237 && pThis->PCHSGeometry.cHeads > 0
238 && pThis->PCHSGeometry.cSectors > 0)
239 {
240 *pPCHSGeometry = pThis->PCHSGeometry;
241 }
242 else
243 rc = VERR_PDM_GEOMETRY_NOT_SET;
244 }
245 else
246 rc = VERR_PDM_MEDIA_NOT_MOUNTED;
247
248 RTCritSectLeave(&pThis->CritSect);
249 LogFlow(("%s-%d: %s: returns %Rrc CHS={%d,%d,%d}\n",
250 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, __FUNCTION__, rc,
251 pThis->PCHSGeometry.cCylinders, pThis->PCHSGeometry.cHeads, pThis->PCHSGeometry.cSectors));
252 return rc;
253}
254
255
256/** @interface_method_impl{PDMIMEDIA,pfnBiosSetPCHSGeometry} */
257static DECLCALLBACK(int) drvHostBaseSetPCHSGeometry(PPDMIMEDIA pInterface, PCPDMMEDIAGEOMETRY pPCHSGeometry)
258{
259 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
260 LogFlow(("%s-%d: %s: cCylinders=%d cHeads=%d cSectors=%d\n",
261 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, __FUNCTION__,
262 pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
263 RTCritSectEnter(&pThis->CritSect);
264
265 int rc = VINF_SUCCESS;
266 if (pThis->fMediaPresent)
267 {
268 pThis->PCHSGeometry = *pPCHSGeometry;
269 }
270 else
271 {
272 AssertMsgFailed(("Invalid state! Not mounted!\n"));
273 rc = VERR_PDM_MEDIA_NOT_MOUNTED;
274 }
275
276 RTCritSectLeave(&pThis->CritSect);
277 return rc;
278}
279
280
281/** @interface_method_impl{PDMIMEDIA,pfnBiosGetLCHSGeometry} */
282static DECLCALLBACK(int) drvHostBaseGetLCHSGeometry(PPDMIMEDIA pInterface, PPDMMEDIAGEOMETRY pLCHSGeometry)
283{
284 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
285 RTCritSectEnter(&pThis->CritSect);
286
287 int rc = VINF_SUCCESS;
288 if (pThis->fMediaPresent)
289 {
290 if ( pThis->LCHSGeometry.cCylinders > 0
291 && pThis->LCHSGeometry.cHeads > 0
292 && pThis->LCHSGeometry.cSectors > 0)
293 {
294 *pLCHSGeometry = pThis->LCHSGeometry;
295 }
296 else
297 rc = VERR_PDM_GEOMETRY_NOT_SET;
298 }
299 else
300 rc = VERR_PDM_MEDIA_NOT_MOUNTED;
301
302 RTCritSectLeave(&pThis->CritSect);
303 LogFlow(("%s-%d: %s: returns %Rrc CHS={%d,%d,%d}\n",
304 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, __FUNCTION__, rc,
305 pThis->LCHSGeometry.cCylinders, pThis->LCHSGeometry.cHeads, pThis->LCHSGeometry.cSectors));
306 return rc;
307}
308
309
310/** @interface_method_impl{PDMIMEDIA,pfnBiosSetLCHSGeometry} */
311static DECLCALLBACK(int) drvHostBaseSetLCHSGeometry(PPDMIMEDIA pInterface, PCPDMMEDIAGEOMETRY pLCHSGeometry)
312{
313 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
314 LogFlow(("%s-%d: %s: cCylinders=%d cHeads=%d cSectors=%d\n",
315 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, __FUNCTION__,
316 pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
317 RTCritSectEnter(&pThis->CritSect);
318
319 int rc = VINF_SUCCESS;
320 if (pThis->fMediaPresent)
321 {
322 pThis->LCHSGeometry = *pLCHSGeometry;
323 }
324 else
325 {
326 AssertMsgFailed(("Invalid state! Not mounted!\n"));
327 rc = VERR_PDM_MEDIA_NOT_MOUNTED;
328 }
329
330 RTCritSectLeave(&pThis->CritSect);
331 return rc;
332}
333
334
335/** @interface_method_impl{PDMIMEDIA,pfnBiosIsVisible} */
336static DECLCALLBACK(bool) drvHostBaseIsVisible(PPDMIMEDIA pInterface)
337{
338 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
339 return pThis->fBiosVisible;
340}
341
342
343/** @interface_method_impl{PDMIMEDIA,pfnGetRegionCount} */
344static DECLCALLBACK(uint32_t) drvHostBaseGetRegionCount(PPDMIMEDIA pInterface)
345{
346 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
347
348 LogFlowFunc(("\n"));
349 uint32_t cRegions = pThis->fMediaPresent ? 1 : 0;
350
351 /* For now just return one region for all devices. */
352 /** @todo Handle CD/DVD passthrough properly. */
353
354 LogFlowFunc(("returns %u\n", cRegions));
355 return cRegions;
356}
357
358/** @interface_method_impl{PDMIMEDIA,pfnQueryRegionProperties} */
359static DECLCALLBACK(int) drvHostBaseQueryRegionProperties(PPDMIMEDIA pInterface, uint32_t uRegion, uint64_t *pu64LbaStart,
360 uint64_t *pcBlocks, uint64_t *pcbBlock,
361 PVDREGIONDATAFORM penmDataForm)
362{
363 LogFlowFunc(("\n"));
364 int rc = VINF_SUCCESS;
365 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
366
367 if (uRegion < 1 && pThis->fMediaPresent)
368 {
369 uint64_t cbMedia;
370 rc = drvHostBaseGetMediaSizeOs(pThis, &cbMedia);
371 if (RT_SUCCESS(rc))
372 {
373 uint64_t cbBlock = 0;
374
375 if (pThis->enmType == PDMMEDIATYPE_DVD)
376 cbBlock = 2048;
377 else
378 cbBlock = 512; /* Floppy. */
379
380 if (pu64LbaStart)
381 *pu64LbaStart = 0;
382 if (pcBlocks)
383 *pcBlocks = cbMedia / cbBlock;
384 if (pcbBlock)
385 *pcbBlock = cbBlock;
386 if (penmDataForm)
387 *penmDataForm = VDREGIONDATAFORM_RAW;
388 }
389 }
390 else
391 rc = VERR_NOT_FOUND;
392
393 LogFlowFunc(("returns %Rrc\n", rc));
394 return rc;
395}
396
397/** @interface_method_impl{PDMIMEDIA,pfnQueryRegionPropertiesForLba} */
398static DECLCALLBACK(int) drvHostBaseQueryRegionPropertiesForLba(PPDMIMEDIA pInterface, uint64_t u64LbaStart,
399 uint32_t *puRegion, uint64_t *pcBlocks,
400 uint64_t *pcbBlock, PVDREGIONDATAFORM penmDataForm)
401{
402 LogFlowFunc(("\n"));
403 int rc = VINF_SUCCESS;
404 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMedia);
405 uint64_t cbMedia;
406 uint64_t cbBlock = 0;
407
408 if (pThis->enmType == PDMMEDIATYPE_DVD)
409 cbBlock = 2048;
410 else
411 cbBlock = 512; /* Floppy. */
412
413 rc = drvHostBaseGetMediaSizeOs(pThis, &cbMedia);
414 if ( RT_SUCCESS(rc)
415 && u64LbaStart < cbMedia / cbBlock)
416 {
417 if (puRegion)
418 *puRegion = 0;
419 if (pcBlocks)
420 *pcBlocks = cbMedia / cbBlock;
421 if (pcbBlock)
422 *pcbBlock = cbBlock;
423 if (penmDataForm)
424 *penmDataForm = VDREGIONDATAFORM_RAW;
425 }
426 else
427 rc = VERR_NOT_FOUND;
428
429 LogFlowFunc(("returns %Rrc\n", rc));
430 return rc;
431}
432
433
434
435/* -=-=-=-=- IMediaEx -=-=-=-=- */
436
437DECLHIDDEN(int) drvHostBaseBufferRetain(PDRVHOSTBASE pThis, PDRVHOSTBASEREQ pReq, size_t cbBuf, bool fWrite, void **ppvBuf)
438{
439 int rc = VINF_SUCCESS;
440
441 if (pThis->cbBuf < cbBuf)
442 {
443 RTMemFree(pThis->pvBuf);
444 pThis->cbBuf = 0;
445 pThis->pvBuf = RTMemAlloc(cbBuf);
446 if (pThis->pvBuf)
447 pThis->cbBuf = cbBuf;
448 else
449 rc = VERR_NO_MEMORY;
450 }
451
452 if (RT_SUCCESS(rc) && fWrite)
453 {
454 RTSGSEG Seg;
455 RTSGBUF SgBuf;
456
457 Seg.pvSeg = pThis->pvBuf;
458 Seg.cbSeg = cbBuf;
459 RTSgBufInit(&SgBuf, &Seg, 1);
460 rc = pThis->pDrvMediaExPort->pfnIoReqCopyToBuf(pThis->pDrvMediaExPort, (PDMMEDIAEXIOREQ)pReq,
461 &pReq->abAlloc[0], 0, &SgBuf, cbBuf);
462 }
463
464 if (RT_SUCCESS(rc))
465 *ppvBuf = pThis->pvBuf;
466
467 return rc;
468}
469
470DECLHIDDEN(int) drvHostBaseBufferRelease(PDRVHOSTBASE pThis, PDRVHOSTBASEREQ pReq, size_t cbBuf, bool fWrite, void *pvBuf)
471{
472 int rc = VINF_SUCCESS;
473
474 if (!fWrite)
475 {
476 RTSGSEG Seg;
477 RTSGBUF SgBuf;
478
479 Seg.pvSeg = pvBuf;
480 Seg.cbSeg = cbBuf;
481 RTSgBufInit(&SgBuf, &Seg, 1);
482 rc = pThis->pDrvMediaExPort->pfnIoReqCopyFromBuf(pThis->pDrvMediaExPort, (PDMMEDIAEXIOREQ)pReq,
483 &pReq->abAlloc[0], 0, &SgBuf, cbBuf);
484 }
485
486 return rc;
487}
488
489/** @interface_method_impl{PDMIMEDIAEX,pfnQueryFeatures} */
490static DECLCALLBACK(int) drvHostBaseQueryFeatures(PPDMIMEDIAEX pInterface, uint32_t *pfFeatures)
491{
492 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMediaEx);
493
494 *pfFeatures = pThis->IMediaEx.pfnIoReqSendScsiCmd ? PDMIMEDIAEX_FEATURE_F_RAWSCSICMD : 0;
495 return VINF_SUCCESS;
496}
497
498/** @interface_method_impl{PDMIMEDIAEX,pfnNotifySuspend} */
499static DECLCALLBACK(void) drvHostBaseNotifySuspend(PPDMIMEDIAEX pInterface)
500{
501 RT_NOREF(pInterface); /* Nothing to do here. */
502}
503
504/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqAllocSizeSet} */
505static DECLCALLBACK(int) drvHostBaseIoReqAllocSizeSet(PPDMIMEDIAEX pInterface, size_t cbIoReqAlloc)
506{
507 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMediaEx);
508
509 pThis->cbIoReqAlloc = RT_UOFFSETOF_DYN(DRVHOSTBASEREQ, abAlloc[cbIoReqAlloc]);
510 return VINF_SUCCESS;
511}
512
513/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqAlloc} */
514static DECLCALLBACK(int) drvHostBaseIoReqAlloc(PPDMIMEDIAEX pInterface, PPDMMEDIAEXIOREQ phIoReq, void **ppvIoReqAlloc,
515 PDMMEDIAEXIOREQID uIoReqId, uint32_t fFlags)
516{
517 RT_NOREF2(uIoReqId, fFlags);
518
519 int rc = VINF_SUCCESS;
520 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMediaEx);
521 PDRVHOSTBASEREQ pReq = (PDRVHOSTBASEREQ)RTMemAllocZ(pThis->cbIoReqAlloc);
522 if (RT_LIKELY(pReq))
523 {
524 pReq->cbReq = 0;
525 pReq->cbResidual = 0;
526 *phIoReq = (PDMMEDIAEXIOREQ)pReq;
527 *ppvIoReqAlloc = &pReq->abAlloc[0];
528 }
529 else
530 rc = VERR_NO_MEMORY;
531
532 return rc;
533}
534
535/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqFree} */
536static DECLCALLBACK(int) drvHostBaseIoReqFree(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq)
537{
538 RT_NOREF1(pInterface);
539 PDRVHOSTBASEREQ pReq = (PDRVHOSTBASEREQ)hIoReq;
540
541 RTMemFree(pReq);
542 return VINF_SUCCESS;
543}
544
545/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqQueryResidual} */
546static DECLCALLBACK(int) drvHostBaseIoReqQueryResidual(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, size_t *pcbResidual)
547{
548 RT_NOREF1(pInterface);
549 PDRVHOSTBASEREQ pReq = (PDRVHOSTBASEREQ)hIoReq;
550
551 *pcbResidual = pReq->cbResidual;
552 return VINF_SUCCESS;
553}
554
555/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqQueryXferSize} */
556static DECLCALLBACK(int) drvHostBaseIoReqQueryXferSize(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, size_t *pcbXfer)
557{
558 RT_NOREF1(pInterface);
559 PDRVHOSTBASEREQ pReq = (PDRVHOSTBASEREQ)hIoReq;
560
561 *pcbXfer = pReq->cbReq;
562 return VINF_SUCCESS;
563}
564
565/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqCancelAll} */
566static DECLCALLBACK(int) drvHostBaseIoReqCancelAll(PPDMIMEDIAEX pInterface)
567{
568 RT_NOREF1(pInterface);
569 return VINF_SUCCESS;
570}
571
572/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqCancel} */
573static DECLCALLBACK(int) drvHostBaseIoReqCancel(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQID uIoReqId)
574{
575 RT_NOREF2(pInterface, uIoReqId);
576 return VERR_PDM_MEDIAEX_IOREQID_NOT_FOUND;
577}
578
579/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqRead} */
580static DECLCALLBACK(int) drvHostBaseIoReqRead(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, uint64_t off, size_t cbRead)
581{
582 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMediaEx);
583 PDRVHOSTBASEREQ pReq = (PDRVHOSTBASEREQ)hIoReq;
584 LogFlow(("%s-%d: drvHostBaseIoReqRead: off=%#llx cbRead=%#x (%s)\n",
585 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, cbRead, pThis->pszDevice));
586 RTCritSectEnter(&pThis->CritSect);
587
588 pReq->cbReq = cbRead;
589 pReq->cbResidual = cbRead;
590
591 STAM_REL_COUNTER_INC(&pThis->StatReqsSubmitted);
592 STAM_REL_COUNTER_INC(&pThis->StatReqsRead);
593
594 /*
595 * Check the state.
596 */
597 int rc;
598 if (pThis->fMediaPresent)
599 {
600 void *pvBuf;
601 rc = drvHostBaseBufferRetain(pThis, pReq, cbRead, false, &pvBuf);
602 if (RT_SUCCESS(rc))
603 {
604 /*
605 * Seek and read.
606 */
607 rc = drvHostBaseReadOs(pThis, off, pvBuf, cbRead);
608 if (RT_SUCCESS(rc))
609 {
610 Log2(("%s-%d: drvHostBaseReadOs: off=%#llx cbRead=%#x\n"
611 "%16.*Rhxd\n",
612 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, cbRead, cbRead, pvBuf));
613
614 pReq->cbResidual = 0;
615 }
616 else
617 Log(("%s-%d: drvHostBaseIoReqRead: drvHostBaseReadOs(%#llx, %p, %#x) -> %Rrc ('%s')\n",
618 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance,
619 off, pvBuf, cbRead, rc, pThis->pszDevice));
620
621 rc = drvHostBaseBufferRelease(pThis, pReq, cbRead, false, pvBuf);
622 }
623 else
624 Log(("%s-%d: drvHostBaseIoReqRead: drvHostBaseBufferRetain(%#llx, %p, %#x) -> %Rrc ('%s')\n",
625 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance,
626 off, pvBuf, cbRead, rc, pThis->pszDevice));
627 }
628 else
629 rc = VERR_MEDIA_NOT_PRESENT;
630
631 if (RT_SUCCESS(rc))
632 {
633 STAM_REL_COUNTER_INC(&pThis->StatReqsSucceeded);
634 STAM_REL_COUNTER_INC(&pThis->StatBytesRead);
635 }
636 else
637 STAM_REL_COUNTER_INC(&pThis->StatReqsFailed);
638
639 RTCritSectLeave(&pThis->CritSect);
640 LogFlow(("%s-%d: drvHostBaseIoReqRead: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
641 return rc;
642}
643
644/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqWrite} */
645static DECLCALLBACK(int) drvHostBaseIoReqWrite(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, uint64_t off, size_t cbWrite)
646{
647 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMediaEx);
648 PDRVHOSTBASEREQ pReq = (PDRVHOSTBASEREQ)hIoReq;
649 LogFlow(("%s-%d: drvHostBaseIoReqWrite: off=%#llx cbWrite=%#x (%s)\n",
650 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, cbWrite, pThis->pszDevice));
651 RTCritSectEnter(&pThis->CritSect);
652
653 pReq->cbReq = cbWrite;
654 pReq->cbResidual = cbWrite;
655
656 STAM_REL_COUNTER_INC(&pThis->StatReqsSubmitted);
657 STAM_REL_COUNTER_INC(&pThis->StatReqsWrite);
658
659 /*
660 * Check the state.
661 */
662 int rc;
663 if (!pThis->fReadOnly)
664 {
665 if (pThis->fMediaPresent)
666 {
667 void *pvBuf;
668 rc = drvHostBaseBufferRetain(pThis, pReq, cbWrite, true, &pvBuf);
669 if (RT_SUCCESS(rc))
670 {
671 Log2(("%s-%d: drvHostBaseIoReqWrite: off=%#llx cbWrite=%#x\n"
672 "%16.*Rhxd\n",
673 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, off, cbWrite, cbWrite, pvBuf));
674 /*
675 * Seek and write.
676 */
677 rc = drvHostBaseWriteOs(pThis, off, pvBuf, cbWrite);
678 if (RT_FAILURE(rc))
679 Log(("%s-%d: drvHostBaseIoReqWrite: drvHostBaseWriteOs(%#llx, %p, %#x) -> %Rrc ('%s')\n",
680 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance,
681 off, pvBuf, cbWrite, rc, pThis->pszDevice));
682 else
683 pReq->cbResidual = 0;
684
685 rc = drvHostBaseBufferRelease(pThis, pReq, cbWrite, true, pvBuf);
686 }
687 }
688 else
689 rc = VERR_MEDIA_NOT_PRESENT;
690 }
691 else
692 rc = VERR_WRITE_PROTECT;
693
694 if (RT_SUCCESS(rc))
695 {
696 STAM_REL_COUNTER_INC(&pThis->StatReqsSucceeded);
697 STAM_REL_COUNTER_INC(&pThis->StatBytesWritten);
698 }
699 else
700 STAM_REL_COUNTER_INC(&pThis->StatReqsFailed);
701
702 RTCritSectLeave(&pThis->CritSect);
703 LogFlow(("%s-%d: drvHostBaseIoReqWrite: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
704 return rc;
705}
706
707/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqFlush} */
708static DECLCALLBACK(int) drvHostBaseIoReqFlush(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq)
709{
710 RT_NOREF1(hIoReq);
711
712 int rc;
713 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMediaEx);
714 LogFlow(("%s-%d: drvHostBaseIoReqFlush: (%s)\n",
715 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDevice));
716 RTCritSectEnter(&pThis->CritSect);
717
718 STAM_REL_COUNTER_INC(&pThis->StatReqsSubmitted);
719 STAM_REL_COUNTER_INC(&pThis->StatReqsFlush);
720
721 if (pThis->fMediaPresent)
722 rc = drvHostBaseFlushOs(pThis);
723 else
724 rc = VERR_MEDIA_NOT_PRESENT;
725
726 if (RT_SUCCESS(rc))
727 STAM_REL_COUNTER_INC(&pThis->StatReqsSucceeded);
728 else
729 STAM_REL_COUNTER_INC(&pThis->StatReqsFailed);
730
731 RTCritSectLeave(&pThis->CritSect);
732 LogFlow(("%s-%d: drvHostBaseFlush: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
733 return rc;
734}
735
736/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqDiscard} */
737static DECLCALLBACK(int) drvHostBaseIoReqDiscard(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, unsigned cRangesMax)
738{
739 RT_NOREF3(pInterface, hIoReq, cRangesMax);
740 return VERR_NOT_SUPPORTED;
741}
742
743/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqGetActiveCount} */
744static DECLCALLBACK(uint32_t) drvHostBaseIoReqGetActiveCount(PPDMIMEDIAEX pInterface)
745{
746 RT_NOREF1(pInterface);
747 return 0;
748}
749
750/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqGetSuspendedCount} */
751static DECLCALLBACK(uint32_t) drvHostBaseIoReqGetSuspendedCount(PPDMIMEDIAEX pInterface)
752{
753 RT_NOREF1(pInterface);
754 return 0;
755}
756
757/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqQuerySuspendedStart} */
758static DECLCALLBACK(int) drvHostBaseIoReqQuerySuspendedStart(PPDMIMEDIAEX pInterface, PPDMMEDIAEXIOREQ phIoReq, void **ppvIoReqAlloc)
759{
760 RT_NOREF3(pInterface, phIoReq, ppvIoReqAlloc);
761 return VERR_NOT_IMPLEMENTED;
762}
763
764/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqQuerySuspendedNext} */
765static DECLCALLBACK(int) drvHostBaseIoReqQuerySuspendedNext(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq,
766 PPDMMEDIAEXIOREQ phIoReqNext, void **ppvIoReqAllocNext)
767{
768 RT_NOREF4(pInterface, hIoReq, phIoReqNext, ppvIoReqAllocNext);
769 return VERR_NOT_IMPLEMENTED;
770}
771
772/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqSuspendedSave} */
773static DECLCALLBACK(int) drvHostBaseIoReqSuspendedSave(PPDMIMEDIAEX pInterface, PSSMHANDLE pSSM, PDMMEDIAEXIOREQ hIoReq)
774{
775 RT_NOREF3(pInterface, pSSM, hIoReq);
776 return VERR_NOT_IMPLEMENTED;
777}
778
779/** @interface_method_impl{PDMIMEDIAEX,pfnIoReqSuspendedLoad} */
780static DECLCALLBACK(int) drvHostBaseIoReqSuspendedLoad(PPDMIMEDIAEX pInterface, PSSMHANDLE pSSM, PDMMEDIAEXIOREQ hIoReq)
781{
782 RT_NOREF3(pInterface, pSSM, hIoReq);
783 return VERR_NOT_IMPLEMENTED;
784}
785
786
787
788/* -=-=-=-=- IMount -=-=-=-=- */
789
790/** @interface_method_impl{PDMIMOUNT,pfnUnmount} */
791static DECLCALLBACK(int) drvHostBaseUnmount(PPDMIMOUNT pInterface, bool fForce, bool fEject)
792{
793 RT_NOREF(fEject);
794 /* While we're not mountable (see drvHostBaseMount), we're unmountable. */
795 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMount);
796 RTCritSectEnter(&pThis->CritSect);
797
798 /*
799 * Validate state.
800 */
801 int rc = VINF_SUCCESS;
802 if (!pThis->fLocked || fForce)
803 {
804 /* Unlock drive if necessary. */
805 if (pThis->fLocked)
806 {
807 if (pThis->pfnDoLock)
808 rc = pThis->pfnDoLock(pThis, false);
809 if (RT_SUCCESS(rc))
810 pThis->fLocked = false;
811 }
812
813 if (fEject)
814 {
815 /*
816 * Eject the disc.
817 */
818 rc = drvHostBaseEjectOs(pThis);
819 }
820
821 /*
822 * Media is no longer present.
823 */
824 DRVHostBaseMediaNotPresent(pThis);
825 }
826 else
827 {
828 Log(("drvHostBaseUnmount: Locked\n"));
829 rc = VERR_PDM_MEDIA_LOCKED;
830 }
831
832 RTCritSectLeave(&pThis->CritSect);
833 LogFlow(("drvHostBaseUnmount: returns %Rrc\n", rc));
834 return rc;
835}
836
837
838/** @interface_method_impl{PDMIMOUNT,pfnIsMounted} */
839static DECLCALLBACK(bool) drvHostBaseIsMounted(PPDMIMOUNT pInterface)
840{
841 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMount);
842 RTCritSectEnter(&pThis->CritSect);
843
844 bool fRc = pThis->fMediaPresent;
845
846 RTCritSectLeave(&pThis->CritSect);
847 return fRc;
848}
849
850
851/** @interface_method_impl{PDMIMOUNT,pfnIsLocked} */
852static DECLCALLBACK(int) drvHostBaseLock(PPDMIMOUNT pInterface)
853{
854 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMount);
855 RTCritSectEnter(&pThis->CritSect);
856
857 int rc = VINF_SUCCESS;
858 if (!pThis->fLocked)
859 {
860 if (pThis->pfnDoLock)
861 rc = pThis->pfnDoLock(pThis, true);
862 if (RT_SUCCESS(rc))
863 pThis->fLocked = true;
864 }
865 else
866 LogFlow(("%s-%d: drvHostBaseLock: already locked\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
867
868 RTCritSectLeave(&pThis->CritSect);
869 LogFlow(("%s-%d: drvHostBaseLock: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
870 return rc;
871}
872
873
874/** @interface_method_impl{PDMIMOUNT,pfnIsLocked} */
875static DECLCALLBACK(int) drvHostBaseUnlock(PPDMIMOUNT pInterface)
876{
877 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMount);
878 RTCritSectEnter(&pThis->CritSect);
879
880 int rc = VINF_SUCCESS;
881 if (pThis->fLocked)
882 {
883 if (pThis->pfnDoLock)
884 rc = pThis->pfnDoLock(pThis, false);
885 if (RT_SUCCESS(rc))
886 pThis->fLocked = false;
887 }
888 else
889 LogFlow(("%s-%d: drvHostBaseUnlock: not locked\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
890
891 RTCritSectLeave(&pThis->CritSect);
892 LogFlow(("%s-%d: drvHostBaseUnlock: returns %Rrc\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, rc));
893 return rc;
894}
895
896
897/** @interface_method_impl{PDMIMOUNT,pfnIsLocked} */
898static DECLCALLBACK(bool) drvHostBaseIsLocked(PPDMIMOUNT pInterface)
899{
900 PDRVHOSTBASE pThis = RT_FROM_MEMBER(pInterface, DRVHOSTBASE, IMount);
901 RTCritSectEnter(&pThis->CritSect);
902
903 bool fRc = pThis->fLocked;
904
905 RTCritSectLeave(&pThis->CritSect);
906 return fRc;
907}
908
909
910/* -=-=-=-=- IBase -=-=-=-=- */
911
912/**
913 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
914 */
915static DECLCALLBACK(void *) drvHostBaseQueryInterface(PPDMIBASE pInterface, const char *pszIID)
916{
917 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
918 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
919
920 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
921 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIA, &pThis->IMedia);
922 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMOUNT, &pThis->IMount);
923 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAEX, pThis->pDrvMediaExPort ? &pThis->IMediaEx : NULL);
924 return NULL;
925}
926
927
928/* -=-=-=-=- poller thread -=-=-=-=- */
929
930
931/**
932 * Media present.
933 * Query the size and notify the above driver / device.
934 *
935 * @param pThis The instance data.
936 */
937DECLHIDDEN(int) DRVHostBaseMediaPresent(PDRVHOSTBASE pThis)
938{
939 /*
940 * Open the drive.
941 */
942 int rc = drvHostBaseMediaRefreshOs(pThis);
943 if (RT_FAILURE(rc))
944 return rc;
945
946 /*
947 * Determine the size.
948 */
949 uint64_t cb;
950 rc = drvHostBaseGetMediaSizeOs(pThis, &cb);
951 if (RT_FAILURE(rc))
952 {
953 LogFlow(("%s-%d: failed to figure media size of %s, rc=%Rrc\n",
954 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->pszDevice, rc));
955 return rc;
956 }
957
958 /*
959 * Update the data and inform the unit.
960 */
961 pThis->cbSize = cb;
962 pThis->fMediaPresent = true;
963 if (pThis->pDrvMountNotify)
964 pThis->pDrvMountNotify->pfnMountNotify(pThis->pDrvMountNotify);
965 LogFlow(("%s-%d: drvHostBaseMediaPresent: cbSize=%lld (%#llx)\n",
966 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, pThis->cbSize, pThis->cbSize));
967 return VINF_SUCCESS;
968}
969
970
971/**
972 * Media no longer present.
973 * @param pThis The instance data.
974 */
975DECLHIDDEN(void) DRVHostBaseMediaNotPresent(PDRVHOSTBASE pThis)
976{
977 pThis->fMediaPresent = false;
978 pThis->fLocked = false;
979 pThis->PCHSGeometry.cCylinders = 0;
980 pThis->PCHSGeometry.cHeads = 0;
981 pThis->PCHSGeometry.cSectors = 0;
982 pThis->LCHSGeometry.cCylinders = 0;
983 pThis->LCHSGeometry.cHeads = 0;
984 pThis->LCHSGeometry.cSectors = 0;
985 if (pThis->pDrvMountNotify)
986 pThis->pDrvMountNotify->pfnUnmountNotify(pThis->pDrvMountNotify);
987}
988
989
990static int drvHostBaseMediaPoll(PDRVHOSTBASE pThis)
991{
992 /*
993 * Poll for media change.
994 */
995 bool fMediaPresent = false;
996 bool fMediaChanged = false;
997 drvHostBaseQueryMediaStatusOs(pThis, &fMediaChanged, &fMediaPresent);
998
999 RTCritSectEnter(&pThis->CritSect);
1000
1001 int rc = VINF_SUCCESS;
1002 if (pThis->fMediaPresent != fMediaPresent)
1003 {
1004 LogFlow(("drvHostDvdPoll: %d -> %d\n", pThis->fMediaPresent, fMediaPresent));
1005 pThis->fMediaPresent = false;
1006 if (fMediaPresent)
1007 rc = DRVHostBaseMediaPresent(pThis);
1008 else
1009 DRVHostBaseMediaNotPresent(pThis);
1010 }
1011 else if (fMediaPresent)
1012 {
1013 /*
1014 * Poll for media change.
1015 */
1016 if (fMediaChanged)
1017 {
1018 LogFlow(("drvHostDVDMediaThread: Media changed!\n"));
1019 DRVHostBaseMediaNotPresent(pThis);
1020 rc = DRVHostBaseMediaPresent(pThis);
1021 }
1022 }
1023
1024 RTCritSectLeave(&pThis->CritSect);
1025 return rc;
1026}
1027
1028
1029/**
1030 * This thread will periodically poll the device for media presence.
1031 *
1032 * @returns Ignored.
1033 * @param ThreadSelf Handle of this thread. Ignored.
1034 * @param pvUser Pointer to the driver instance structure.
1035 */
1036static DECLCALLBACK(int) drvHostBaseMediaThread(RTTHREAD ThreadSelf, void *pvUser)
1037{
1038 PDRVHOSTBASE pThis = (PDRVHOSTBASE)pvUser;
1039 LogFlow(("%s-%d: drvHostBaseMediaThread: ThreadSelf=%p pvUser=%p\n",
1040 pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance, ThreadSelf, pvUser));
1041 bool fFirst = true;
1042 int cRetries = 10;
1043 while (!pThis->fShutdownPoller)
1044 {
1045 /*
1046 * Perform the polling (unless we've run out of 50ms retries).
1047 */
1048 if (cRetries-- > 0)
1049 {
1050
1051 int rc = drvHostBaseMediaPoll(pThis);
1052 if (RT_FAILURE(rc))
1053 {
1054 RTSemEventWait(pThis->EventPoller, 50);
1055 continue;
1056 }
1057 }
1058
1059 /*
1060 * Signal EMT after the first go.
1061 */
1062 if (fFirst)
1063 {
1064 RTThreadUserSignal(ThreadSelf);
1065 fFirst = false;
1066 }
1067
1068 /*
1069 * Sleep.
1070 */
1071 int rc = RTSemEventWait(pThis->EventPoller, pThis->cMilliesPoller);
1072 if ( RT_FAILURE(rc)
1073 && rc != VERR_TIMEOUT)
1074 {
1075 AssertMsgFailed(("rc=%Rrc\n", rc));
1076 pThis->ThreadPoller = NIL_RTTHREAD;
1077 LogFlow(("drvHostBaseMediaThread: returns %Rrc\n", rc));
1078 return rc;
1079 }
1080 cRetries = 10;
1081 }
1082
1083 /* (Don't clear the thread handle here, the destructor thread is using it to wait.) */
1084 LogFlow(("%s-%d: drvHostBaseMediaThread: returns VINF_SUCCESS\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
1085 return VINF_SUCCESS;
1086}
1087
1088/**
1089 * Registers statistics associated with the given media driver.
1090 *
1091 * @returns VBox status code.
1092 * @param pThis The media driver instance.
1093 */
1094static int drvHostBaseStatsRegister(PDRVHOSTBASE pThis)
1095{
1096 PPDMDRVINS pDrvIns = pThis->pDrvIns;
1097 uint32_t iInstance, iLUN;
1098 const char *pcszController;
1099
1100 int rc = pThis->pDrvMediaPort->pfnQueryDeviceLocation(pThis->pDrvMediaPort, &pcszController,
1101 &iInstance, &iLUN);
1102 if (RT_SUCCESS(rc))
1103 {
1104 char *pszCtrlUpper = RTStrDup(pcszController);
1105 if (pszCtrlUpper)
1106 {
1107 RTStrToUpper(pszCtrlUpper);
1108
1109 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatBytesRead, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_BYTES,
1110 "Amount of data read.", "/Devices/%s%u/Port%u/ReadBytes", pszCtrlUpper, iInstance, iLUN);
1111 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatBytesWritten, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_BYTES,
1112 "Amount of data written.", "/Devices/%s%u/Port%u/WrittenBytes", pszCtrlUpper, iInstance, iLUN);
1113
1114 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatReqsSubmitted, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_COUNT,
1115 "Number of I/O requests submitted.", "/Devices/%s%u/Port%u/ReqsSubmitted", pszCtrlUpper, iInstance, iLUN);
1116 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatReqsFailed, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_COUNT,
1117 "Number of I/O requests failed.", "/Devices/%s%u/Port%u/ReqsFailed", pszCtrlUpper, iInstance, iLUN);
1118 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatReqsSucceeded, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_COUNT,
1119 "Number of I/O requests succeeded.", "/Devices/%s%u/Port%u/ReqsSucceeded", pszCtrlUpper, iInstance, iLUN);
1120 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatReqsFlush, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_COUNT,
1121 "Number of flush I/O requests submitted.", "/Devices/%s%u/Port%u/ReqsFlush", pszCtrlUpper, iInstance, iLUN);
1122 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatReqsWrite, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_COUNT,
1123 "Number of write I/O requests submitted.", "/Devices/%s%u/Port%u/ReqsWrite", pszCtrlUpper, iInstance, iLUN);
1124 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatReqsRead, STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_COUNT,
1125 "Number of read I/O requests submitted.", "/Devices/%s%u/Port%u/ReqsRead", pszCtrlUpper, iInstance, iLUN);
1126
1127 RTStrFree(pszCtrlUpper);
1128 }
1129 else
1130 rc = VERR_NO_STR_MEMORY;
1131 }
1132
1133 return rc;
1134}
1135
1136/**
1137 * Deregisters statistics associated with the given media driver.
1138 *
1139 * @returns nothing.
1140 * @param pThis The media driver instance.
1141 */
1142static void drvhostBaseStatsDeregister(PDRVHOSTBASE pThis)
1143{
1144 PPDMDRVINS pDrvIns = pThis->pDrvIns;
1145
1146 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatBytesRead);
1147 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatBytesWritten);
1148 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReqsSubmitted);
1149 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReqsFailed);
1150 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReqsSucceeded);
1151 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReqsFlush);
1152 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReqsWrite);
1153 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReqsRead);
1154}
1155
1156/* -=-=-=-=- driver interface -=-=-=-=- */
1157
1158
1159/**
1160 * Done state load operation.
1161 *
1162 * @returns VBox load code.
1163 * @param pDrvIns Driver instance of the driver which registered the data unit.
1164 * @param pSSM SSM operation handle.
1165 */
1166static DECLCALLBACK(int) drvHostBaseLoadDone(PPDMDRVINS pDrvIns, PSSMHANDLE pSSM)
1167{
1168 RT_NOREF(pSSM);
1169 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
1170 LogFlow(("%s-%d: drvHostBaseMediaThread:\n", pThis->pDrvIns->pReg->szName, pThis->pDrvIns->iInstance));
1171 RTCritSectEnter(&pThis->CritSect);
1172
1173 /*
1174 * Tell the device/driver above us that the media status is uncertain.
1175 */
1176 if (pThis->pDrvMountNotify)
1177 {
1178 pThis->pDrvMountNotify->pfnUnmountNotify(pThis->pDrvMountNotify);
1179 if (pThis->fMediaPresent)
1180 pThis->pDrvMountNotify->pfnMountNotify(pThis->pDrvMountNotify);
1181 }
1182
1183 RTCritSectLeave(&pThis->CritSect);
1184 return VINF_SUCCESS;
1185}
1186
1187
1188/** @copydoc FNPDMDRVDESTRUCT */
1189DECLCALLBACK(void) DRVHostBaseDestruct(PPDMDRVINS pDrvIns)
1190{
1191 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
1192 LogFlow(("%s-%d: drvHostBaseDestruct: iInstance=%d\n", pDrvIns->pReg->szName, pDrvIns->iInstance, pDrvIns->iInstance));
1193
1194 /*
1195 * Terminate the thread.
1196 */
1197 if (pThis->ThreadPoller != NIL_RTTHREAD)
1198 {
1199 pThis->fShutdownPoller = true;
1200 int rc;
1201 int cTimes = 50;
1202 do
1203 {
1204 RTSemEventSignal(pThis->EventPoller);
1205 rc = RTThreadWait(pThis->ThreadPoller, 100, NULL);
1206 } while (cTimes-- > 0 && rc == VERR_TIMEOUT);
1207
1208 if (!rc)
1209 pThis->ThreadPoller = NIL_RTTHREAD;
1210 }
1211
1212 /*
1213 * Cleanup the other resources.
1214 */
1215 drvHostBaseDestructOs(pThis);
1216
1217 if (pThis->EventPoller != NULL)
1218 {
1219 RTSemEventDestroy(pThis->EventPoller);
1220 pThis->EventPoller = NULL;
1221 }
1222
1223 if (pThis->pszDevice)
1224 {
1225 MMR3HeapFree(pThis->pszDevice);
1226 pThis->pszDevice = NULL;
1227 }
1228
1229 if (pThis->pszDeviceOpen)
1230 {
1231 RTStrFree(pThis->pszDeviceOpen);
1232 pThis->pszDeviceOpen = NULL;
1233 }
1234
1235 if (pThis->pvBuf)
1236 {
1237 RTMemFree(pThis->pvBuf);
1238 pThis->pvBuf = NULL;
1239 pThis->cbBuf = 0;
1240 }
1241
1242 /* Forget about the notifications. */
1243 pThis->pDrvMountNotify = NULL;
1244
1245 drvhostBaseStatsDeregister(pThis);
1246
1247 /* Leave the instance operational if this is just a cleanup of the state
1248 * after an attach error happened. So don't destroy the critsect then. */
1249 if (!pThis->fKeepInstance && RTCritSectIsInitialized(&pThis->CritSect))
1250 RTCritSectDelete(&pThis->CritSect);
1251 LogFlow(("%s-%d: drvHostBaseDestruct completed\n", pDrvIns->pReg->szName, pDrvIns->iInstance));
1252}
1253
1254
1255/**
1256 * Initializes the instance data .
1257 *
1258 * On failure call DRVHostBaseDestruct().
1259 *
1260 * @returns VBox status code.
1261 * @param pDrvIns Driver instance.
1262 * @param pszCfgValid Pointer to a string of valid CFGM options.
1263 * @param pCfg Configuration handle.
1264 * @param enmType Device type.
1265 */
1266DECLHIDDEN(int) DRVHostBaseInit(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, const char *pszCfgValid, PDMMEDIATYPE enmType)
1267{
1268 int src = VINF_SUCCESS;
1269 PDRVHOSTBASE pThis = PDMINS_2_DATA(pDrvIns, PDRVHOSTBASE);
1270 LogFlow(("%s-%d: DRVHostBaseInit: iInstance=%d\n", pDrvIns->pReg->szName, pDrvIns->iInstance, pDrvIns->iInstance));
1271
1272 /*
1273 * Initialize most of the data members.
1274 */
1275 pThis->pDrvIns = pDrvIns;
1276 pThis->fKeepInstance = false;
1277 pThis->ThreadPoller = NIL_RTTHREAD;
1278 pThis->enmType = enmType;
1279 pThis->fAttachFailError = true; /* It's an error until we've read the config. */
1280
1281 /* IBase. */
1282 pDrvIns->IBase.pfnQueryInterface = drvHostBaseQueryInterface;
1283
1284 /* IMedia. */
1285 pThis->IMedia.pfnRead = drvHostBaseRead;
1286 pThis->IMedia.pfnWrite = drvHostBaseWrite;
1287 pThis->IMedia.pfnFlush = drvHostBaseFlush;
1288 pThis->IMedia.pfnIsReadOnly = drvHostBaseIsReadOnly;
1289 pThis->IMedia.pfnIsNonRotational = drvHostBaseIsNonRotational;
1290 pThis->IMedia.pfnGetSize = drvHostBaseGetSize;
1291 pThis->IMedia.pfnGetType = drvHostBaseGetType;
1292 pThis->IMedia.pfnGetUuid = drvHostBaseGetUuid;
1293 pThis->IMedia.pfnBiosGetPCHSGeometry = drvHostBaseGetPCHSGeometry;
1294 pThis->IMedia.pfnBiosSetPCHSGeometry = drvHostBaseSetPCHSGeometry;
1295 pThis->IMedia.pfnBiosGetLCHSGeometry = drvHostBaseGetLCHSGeometry;
1296 pThis->IMedia.pfnBiosSetLCHSGeometry = drvHostBaseSetLCHSGeometry;
1297 pThis->IMedia.pfnBiosIsVisible = drvHostBaseIsVisible;
1298 pThis->IMedia.pfnGetRegionCount = drvHostBaseGetRegionCount;
1299 pThis->IMedia.pfnQueryRegionProperties = drvHostBaseQueryRegionProperties;
1300 pThis->IMedia.pfnQueryRegionPropertiesForLba = drvHostBaseQueryRegionPropertiesForLba;
1301
1302 /* IMediaEx */
1303 pThis->IMediaEx.pfnQueryFeatures = drvHostBaseQueryFeatures;
1304 pThis->IMediaEx.pfnNotifySuspend = drvHostBaseNotifySuspend;
1305 pThis->IMediaEx.pfnIoReqAllocSizeSet = drvHostBaseIoReqAllocSizeSet;
1306 pThis->IMediaEx.pfnIoReqAlloc = drvHostBaseIoReqAlloc;
1307 pThis->IMediaEx.pfnIoReqFree = drvHostBaseIoReqFree;
1308 pThis->IMediaEx.pfnIoReqQueryResidual = drvHostBaseIoReqQueryResidual;
1309 pThis->IMediaEx.pfnIoReqQueryXferSize = drvHostBaseIoReqQueryXferSize;
1310 pThis->IMediaEx.pfnIoReqCancelAll = drvHostBaseIoReqCancelAll;
1311 pThis->IMediaEx.pfnIoReqCancel = drvHostBaseIoReqCancel;
1312 pThis->IMediaEx.pfnIoReqRead = drvHostBaseIoReqRead;
1313 pThis->IMediaEx.pfnIoReqWrite = drvHostBaseIoReqWrite;
1314 pThis->IMediaEx.pfnIoReqFlush = drvHostBaseIoReqFlush;
1315 pThis->IMediaEx.pfnIoReqDiscard = drvHostBaseIoReqDiscard;
1316 pThis->IMediaEx.pfnIoReqGetActiveCount = drvHostBaseIoReqGetActiveCount;
1317 pThis->IMediaEx.pfnIoReqGetSuspendedCount = drvHostBaseIoReqGetSuspendedCount;
1318 pThis->IMediaEx.pfnIoReqQuerySuspendedStart = drvHostBaseIoReqQuerySuspendedStart;
1319 pThis->IMediaEx.pfnIoReqQuerySuspendedNext = drvHostBaseIoReqQuerySuspendedNext;
1320 pThis->IMediaEx.pfnIoReqSuspendedSave = drvHostBaseIoReqSuspendedSave;
1321 pThis->IMediaEx.pfnIoReqSuspendedLoad = drvHostBaseIoReqSuspendedLoad;
1322
1323 /* IMount. */
1324 pThis->IMount.pfnUnmount = drvHostBaseUnmount;
1325 pThis->IMount.pfnIsMounted = drvHostBaseIsMounted;
1326 pThis->IMount.pfnLock = drvHostBaseLock;
1327 pThis->IMount.pfnUnlock = drvHostBaseUnlock;
1328 pThis->IMount.pfnIsLocked = drvHostBaseIsLocked;
1329
1330 drvHostBaseInitOs(pThis);
1331
1332 if (!CFGMR3AreValuesValid(pCfg, pszCfgValid))
1333 {
1334 pThis->fAttachFailError = true;
1335 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
1336 }
1337
1338 /*
1339 * Get the IMediaPort & IMountNotify interfaces of the above driver/device.
1340 */
1341 pThis->pDrvMediaPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIMEDIAPORT);
1342 if (!pThis->pDrvMediaPort)
1343 {
1344 AssertMsgFailed(("Configuration error: No media port interface above!\n"));
1345 return VERR_PDM_MISSING_INTERFACE_ABOVE;
1346 }
1347 pThis->pDrvMediaExPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIMEDIAEXPORT);
1348 pThis->pDrvMountNotify = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIMOUNTNOTIFY);
1349
1350 /*
1351 * Query configuration.
1352 */
1353 /* Device */
1354 int rc = CFGMR3QueryStringAlloc(pCfg, "Path", &pThis->pszDevice);
1355 if (RT_FAILURE(rc))
1356 {
1357 AssertMsgFailed(("Configuration error: query for \"Path\" string returned %Rra.\n", rc));
1358 return rc;
1359 }
1360
1361 /* Mountable */
1362 uint32_t u32;
1363 rc = CFGMR3QueryU32Def(pCfg, "Interval", &u32, 1000);
1364 if (RT_SUCCESS(rc))
1365 pThis->cMilliesPoller = u32;
1366 else
1367 {
1368 AssertMsgFailed(("Configuration error: Query \"Mountable\" resulted in %Rrc.\n", rc));
1369 return rc;
1370 }
1371
1372 /* ReadOnly - passthrough mode requires read/write access in any case. */
1373 if ( (pThis->enmType == PDMMEDIATYPE_CDROM || pThis->enmType == PDMMEDIATYPE_DVD)
1374 && pThis->IMedia.pfnSendCmd)
1375 pThis->fReadOnlyConfig = false;
1376 else
1377 {
1378 rc = CFGMR3QueryBoolDef(pCfg, "ReadOnly", &pThis->fReadOnlyConfig,
1379 enmType == PDMMEDIATYPE_DVD || enmType == PDMMEDIATYPE_CDROM
1380 ? true
1381 : false);
1382 if (RT_FAILURE(rc))
1383 {
1384 AssertMsgFailed(("Configuration error: Query \"ReadOnly\" resulted in %Rrc.\n", rc));
1385 return rc;
1386 }
1387 }
1388
1389 /* Locked */
1390 rc = CFGMR3QueryBoolDef(pCfg, "Locked", &pThis->fLocked, false);
1391 if (RT_FAILURE(rc))
1392 {
1393 AssertMsgFailed(("Configuration error: Query \"Locked\" resulted in %Rrc.\n", rc));
1394 return rc;
1395 }
1396
1397 /* BIOS visible */
1398 rc = CFGMR3QueryBoolDef(pCfg, "BIOSVisible", &pThis->fBiosVisible, true);
1399 if (RT_FAILURE(rc))
1400 {
1401 AssertMsgFailed(("Configuration error: Query \"BIOSVisible\" resulted in %Rrc.\n", rc));
1402 return rc;
1403 }
1404
1405 /* Uuid */
1406 char *psz;
1407 rc = CFGMR3QueryStringAlloc(pCfg, "Uuid", &psz);
1408 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1409 RTUuidClear(&pThis->Uuid);
1410 else if (RT_SUCCESS(rc))
1411 {
1412 rc = RTUuidFromStr(&pThis->Uuid, psz);
1413 if (RT_FAILURE(rc))
1414 {
1415 AssertMsgFailed(("Configuration error: Uuid from string failed on \"%s\", rc=%Rrc.\n", psz, rc));
1416 MMR3HeapFree(psz);
1417 return rc;
1418 }
1419 MMR3HeapFree(psz);
1420 }
1421 else
1422 {
1423 AssertMsgFailed(("Configuration error: Failed to obtain the uuid, rc=%Rrc.\n", rc));
1424 return rc;
1425 }
1426
1427 /* Define whether attach failure is an error (default) or not. */
1428 bool fAttachFailError = true;
1429 rc = CFGMR3QueryBoolDef(pCfg, "AttachFailError", &fAttachFailError, true);
1430 pThis->fAttachFailError = fAttachFailError;
1431
1432 /* log config summary */
1433 Log(("%s-%d: pszDevice='%s' (%s) cMilliesPoller=%d fReadOnlyConfig=%d fLocked=%d fBIOSVisible=%d Uuid=%RTuuid\n",
1434 pDrvIns->pReg->szName, pDrvIns->iInstance, pThis->pszDevice, pThis->pszDeviceOpen, pThis->cMilliesPoller,
1435 pThis->fReadOnlyConfig, pThis->fLocked, pThis->fBiosVisible, &pThis->Uuid));
1436
1437 /*
1438 * Check that there are no drivers below us.
1439 */
1440 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
1441 ("Configuration error: Not possible to attach anything to this driver!\n"),
1442 VERR_PDM_DRVINS_NO_ATTACH);
1443
1444 /*
1445 * Register saved state.
1446 */
1447 rc = PDMDrvHlpSSMRegisterLoadDone(pDrvIns, drvHostBaseLoadDone);
1448 if (RT_FAILURE(rc))
1449 return rc;
1450
1451 /*
1452 * Initialize the critical section used for serializing the access to the media.
1453 */
1454 rc = RTCritSectInit(&pThis->CritSect);
1455 if (RT_FAILURE(rc))
1456 return rc;
1457
1458 /*
1459 * Open the device.
1460 */
1461 rc = drvHostBaseOpenOs(pThis, pThis->fReadOnlyConfig);
1462 if (RT_FAILURE(rc))
1463 {
1464 char *pszDevice = pThis->pszDevice;
1465#ifndef RT_OS_DARWIN
1466 char szPathReal[256];
1467 if ( RTPathExists(pszDevice)
1468 && RT_SUCCESS(RTPathReal(pszDevice, szPathReal, sizeof(szPathReal))))
1469 pszDevice = szPathReal;
1470#endif
1471
1472 /*
1473 * Disable CD/DVD passthrough in case it was enabled. Would cause
1474 * weird failures later when the guest issues commands. These would
1475 * all fail because of the invalid file handle. So use the normal
1476 * virtual CD/DVD code, which deals more gracefully with unavailable
1477 * "media" - actually a complete drive in this case.
1478 */
1479 pThis->IMedia.pfnSendCmd = NULL;
1480 AssertMsgFailed(("Could not open host device %s, rc=%Rrc\n", pszDevice, rc));
1481 switch (rc)
1482 {
1483 case VERR_ACCESS_DENIED:
1484 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1485#ifdef RT_OS_LINUX
1486 N_("Cannot open host device '%s' for %s access. Check the permissions "
1487 "of that device ('/bin/ls -l %s'): Most probably you need to be member "
1488 "of the device group. Make sure that you logout/login after changing "
1489 "the group settings of the current user"),
1490#else
1491 N_("Cannot open host device '%s' for %s access. Check the permissions "
1492 "of that device"),
1493#endif
1494 pszDevice, pThis->fReadOnlyConfig ? "readonly" : "read/write",
1495 pszDevice);
1496 default:
1497 {
1498 if (pThis->fAttachFailError)
1499 return rc;
1500 int erc = PDMDrvHlpVMSetRuntimeError(pDrvIns, 0 /*fFlags*/,
1501 "DrvHost_MOUNTFAIL",
1502 N_("Cannot attach to host device '%s'"), pszDevice);
1503 AssertRC(erc);
1504 src = rc;
1505 }
1506 }
1507 }
1508
1509 /*
1510 * Lock the drive if that's required by the configuration.
1511 */
1512 if (pThis->fLocked)
1513 {
1514 if (pThis->pfnDoLock)
1515 rc = pThis->pfnDoLock(pThis, true);
1516 if (RT_FAILURE(rc))
1517 {
1518 AssertMsgFailed(("Failed to lock the dvd drive. rc=%Rrc\n", rc));
1519 return rc;
1520 }
1521 }
1522
1523 if (RT_SUCCESS(src) && drvHostBaseIsMediaPollingRequiredOs(pThis))
1524 {
1525 /*
1526 * Create the event semaphore which the poller thread will wait on.
1527 */
1528 rc = RTSemEventCreate(&pThis->EventPoller);
1529 if (RT_FAILURE(rc))
1530 return rc;
1531
1532 /*
1533 * Start the thread which will poll for the media.
1534 */
1535 rc = RTThreadCreate(&pThis->ThreadPoller, drvHostBaseMediaThread, pThis, 0,
1536 RTTHREADTYPE_INFREQUENT_POLLER, RTTHREADFLAGS_WAITABLE, "DVDMEDIA");
1537 if (RT_FAILURE(rc))
1538 {
1539 AssertMsgFailed(("Failed to create poller thread. rc=%Rrc\n", rc));
1540 return rc;
1541 }
1542
1543 /*
1544 * Wait for the thread to start up (!w32:) and do one detection loop.
1545 */
1546 rc = RTThreadUserWait(pThis->ThreadPoller, 10000);
1547 AssertRC(rc);
1548 }
1549
1550 if (RT_SUCCESS(rc))
1551 drvHostBaseStatsRegister(pThis);
1552
1553 if (RT_FAILURE(rc))
1554 {
1555 if (!pThis->fAttachFailError)
1556 {
1557 /* Suppressing the attach failure error must not affect the normal
1558 * DRVHostBaseDestruct, so reset this flag below before leaving. */
1559 pThis->fKeepInstance = true;
1560 rc = VINF_SUCCESS;
1561 }
1562 DRVHostBaseDestruct(pDrvIns);
1563 pThis->fKeepInstance = false;
1564 }
1565
1566 if (RT_FAILURE(src))
1567 return src;
1568
1569 return rc;
1570}
1571
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