VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvBlock.cpp@ 52345

Last change on this file since 52345 was 52023, checked in by vboxsync, 10 years ago

pdmifs.h,DrvBlock,DrvVD: Add methods to allocate and free I/O buffer memory to PDMIBLOCK and PDMIMEDIA. DrvVD decides how to allocate the memory based on the setup

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 38.3 KB
Line 
1/* $Id: DrvBlock.cpp 52023 2014-07-14 21:00:18Z vboxsync $ */
2/** @file
3 * VBox storage devices: Generic block driver
4 */
5
6/*
7 * Copyright (C) 2006-2012 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_BLOCK
23#include <VBox/vmm/pdmdrv.h>
24#include <iprt/assert.h>
25#include <iprt/string.h>
26#include <iprt/uuid.h>
27
28#include "VBoxDD.h"
29
30
31/** @def VBOX_PERIODIC_FLUSH
32 * Enable support for periodically flushing the VDI to disk. This may prove
33 * useful for those nasty problems with the ultra-slow host filesystems.
34 * If this is enabled, it can be configured via the CFGM key
35 * "VBoxInternal/Devices/piix3ide/0/LUN#<x>/Config/FlushInterval". <x>
36 * must be replaced with the correct LUN number of the disk that should
37 * do the periodic flushes. The value of the key is the number of bytes
38 * written between flushes. A value of 0 (the default) denotes no flushes. */
39#define VBOX_PERIODIC_FLUSH
40
41/** @def VBOX_IGNORE_FLUSH
42 * Enable support for ignoring VDI flush requests. This can be useful for
43 * filesystems that show bad guest IDE write performance (especially with
44 * Windows guests). NOTE that this does not disable the flushes caused by
45 * the periodic flush cache feature above.
46 * If this feature is enabled, it can be configured via the CFGM key
47 * "VBoxInternal/Devices/piix3ide/0/LUN#<x>/Config/IgnoreFlush". <x>
48 * must be replaced with the correct LUN number of the disk that should
49 * ignore flush requests. The value of the key is a boolean. The default
50 * is to ignore flushes, i.e. true. */
51#define VBOX_IGNORE_FLUSH
52
53
54/*******************************************************************************
55* Structures and Typedefs *
56*******************************************************************************/
57/**
58 * Block driver instance data.
59 *
60 * @implements PDMIBLOCK
61 * @implements PDMIBLOCKBIOS
62 * @implements PDMIMOUNT
63 * @implements PDMIMEDIAASYNCPORT
64 * @implements PDMIBLOCKASYNC
65 */
66typedef struct DRVBLOCK
67{
68 /** Pointer driver instance. */
69 PPDMDRVINS pDrvIns;
70 /** Drive type. */
71 PDMBLOCKTYPE enmType;
72 /** Locked indicator. */
73 bool fLocked;
74 /** Mountable indicator. */
75 bool fMountable;
76 /** Visible to the BIOS. */
77 bool fBiosVisible;
78#ifdef VBOX_PERIODIC_FLUSH
79 /** HACK: Configuration value for number of bytes written after which to flush. */
80 uint32_t cbFlushInterval;
81 /** HACK: Current count for the number of bytes written since the last flush. */
82 uint32_t cbDataWritten;
83#endif /* VBOX_PERIODIC_FLUSH */
84#ifdef VBOX_IGNORE_FLUSH
85 /** HACK: Disable flushes for this drive. */
86 bool fIgnoreFlush;
87 /** Disable async flushes for this drive. */
88 bool fIgnoreFlushAsync;
89#endif /* VBOX_IGNORE_FLUSH */
90 /** Pointer to the media driver below us.
91 * This is NULL if the media is not mounted. */
92 PPDMIMEDIA pDrvMedia;
93 /** Pointer to the block port interface above us. */
94 PPDMIBLOCKPORT pDrvBlockPort;
95 /** Pointer to the mount notify interface above us. */
96 PPDMIMOUNTNOTIFY pDrvMountNotify;
97 /** Our block interface. */
98 PDMIBLOCK IBlock;
99 /** Our block interface. */
100 PDMIBLOCKBIOS IBlockBios;
101 /** Our mountable interface. */
102 PDMIMOUNT IMount;
103 /** Our media port interface. */
104 PDMIMEDIAPORT IMediaPort;
105
106 /** Pointer to the async media driver below us.
107 * This is NULL if the media is not mounted. */
108 PPDMIMEDIAASYNC pDrvMediaAsync;
109 /** Our media async port. */
110 PDMIMEDIAASYNCPORT IMediaAsyncPort;
111 /** Pointer to the async block port interface above us. */
112 PPDMIBLOCKASYNCPORT pDrvBlockAsyncPort;
113 /** Our async block interface. */
114 PDMIBLOCKASYNC IBlockAsync;
115
116 /** Uuid of the drive. */
117 RTUUID Uuid;
118
119 /** BIOS PCHS Geometry. */
120 PDMMEDIAGEOMETRY PCHSGeometry;
121 /** BIOS LCHS Geometry. */
122 PDMMEDIAGEOMETRY LCHSGeometry;
123} DRVBLOCK, *PDRVBLOCK;
124
125
126/* -=-=-=-=- IBlock -=-=-=-=- */
127
128/** Makes a PDRVBLOCK out of a PPDMIBLOCK. */
129#define PDMIBLOCK_2_DRVBLOCK(pInterface) ( (PDRVBLOCK)((uintptr_t)pInterface - RT_OFFSETOF(DRVBLOCK, IBlock)) )
130
131/** @copydoc PDMIBLOCK::pfnRead */
132static DECLCALLBACK(int) drvblockRead(PPDMIBLOCK pInterface, uint64_t off, void *pvBuf, size_t cbRead)
133{
134 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
135
136 /*
137 * Check the state.
138 */
139 if (!pThis->pDrvMedia)
140 {
141 AssertMsgFailed(("Invalid state! Not mounted!\n"));
142 return VERR_PDM_MEDIA_NOT_MOUNTED;
143 }
144
145 int rc = pThis->pDrvMedia->pfnRead(pThis->pDrvMedia, off, pvBuf, cbRead);
146 return rc;
147}
148
149
150/** @copydoc PDMIBLOCK::pfnWrite */
151static DECLCALLBACK(int) drvblockWrite(PPDMIBLOCK pInterface, uint64_t off, const void *pvBuf, size_t cbWrite)
152{
153 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
154
155 /*
156 * Check the state.
157 */
158 if (!pThis->pDrvMedia)
159 {
160 AssertMsgFailed(("Invalid state! Not mounted!\n"));
161 return VERR_PDM_MEDIA_NOT_MOUNTED;
162 }
163
164 /* Set an FTM checkpoint as this operation changes the state permanently. */
165 PDMDrvHlpFTSetCheckpoint(pThis->pDrvIns, FTMCHECKPOINTTYPE_STORAGE);
166
167 int rc = pThis->pDrvMedia->pfnWrite(pThis->pDrvMedia, off, pvBuf, cbWrite);
168#ifdef VBOX_PERIODIC_FLUSH
169 if (pThis->cbFlushInterval)
170 {
171 pThis->cbDataWritten += (uint32_t)cbWrite;
172 if (pThis->cbDataWritten > pThis->cbFlushInterval)
173 {
174 pThis->cbDataWritten = 0;
175 pThis->pDrvMedia->pfnFlush(pThis->pDrvMedia);
176 }
177 }
178#endif /* VBOX_PERIODIC_FLUSH */
179
180 return rc;
181}
182
183
184/** @copydoc PDMIBLOCK::pfnFlush */
185static DECLCALLBACK(int) drvblockFlush(PPDMIBLOCK pInterface)
186{
187 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
188
189 /*
190 * Check the state.
191 */
192 if (!pThis->pDrvMedia)
193 {
194 AssertMsgFailed(("Invalid state! Not mounted!\n"));
195 return VERR_PDM_MEDIA_NOT_MOUNTED;
196 }
197
198#ifdef VBOX_IGNORE_FLUSH
199 if (pThis->fIgnoreFlush)
200 return VINF_SUCCESS;
201#endif /* VBOX_IGNORE_FLUSH */
202
203 int rc = pThis->pDrvMedia->pfnFlush(pThis->pDrvMedia);
204 if (rc == VERR_NOT_IMPLEMENTED)
205 rc = VINF_SUCCESS;
206 return rc;
207}
208
209
210/** @copydoc PDMIBLOCK::pfnMerge */
211static DECLCALLBACK(int) drvblockMerge(PPDMIBLOCK pInterface,
212 PFNSIMPLEPROGRESS pfnProgress,
213 void *pvUser)
214{
215 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
216
217 /*
218 * Check the state.
219 */
220 if (!pThis->pDrvMedia)
221 {
222 AssertMsgFailed(("Invalid state! Not mounted!\n"));
223 return VERR_PDM_MEDIA_NOT_MOUNTED;
224 }
225
226 if (!pThis->pDrvMedia->pfnMerge)
227 return VERR_NOT_SUPPORTED;
228
229 int rc = pThis->pDrvMedia->pfnMerge(pThis->pDrvMedia, pfnProgress, pvUser);
230 return rc;
231}
232
233
234/** @copydoc PDMIBLOCK::pfnIsReadOnly */
235static DECLCALLBACK(bool) drvblockIsReadOnly(PPDMIBLOCK pInterface)
236{
237 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
238
239 /*
240 * Check the state.
241 */
242 if (!pThis->pDrvMedia)
243 return false;
244
245 bool fRc = pThis->pDrvMedia->pfnIsReadOnly(pThis->pDrvMedia);
246 return fRc;
247}
248
249
250/** @copydoc PDMIBLOCK::pfnGetSize */
251static DECLCALLBACK(uint64_t) drvblockGetSize(PPDMIBLOCK pInterface)
252{
253 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
254
255 /*
256 * Check the state.
257 */
258 if (!pThis->pDrvMedia)
259 return 0;
260
261 uint64_t cb = pThis->pDrvMedia->pfnGetSize(pThis->pDrvMedia);
262 LogFlow(("drvblockGetSize: returns %llu\n", cb));
263 return cb;
264}
265
266
267/** @copydoc PDMIBLOCK::pfnGetSize */
268static DECLCALLBACK(uint32_t) drvblockGetSectorSize(PPDMIBLOCK pInterface)
269{
270 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
271
272 /*
273 * Check the state.
274 */
275 if (!pThis->pDrvMedia)
276 return 0;
277
278 uint32_t cb = pThis->pDrvMedia->pfnGetSectorSize(pThis->pDrvMedia);
279 LogFlowFunc(("returns %u\n", cb));
280 return cb;
281}
282
283
284/** @copydoc PDMIBLOCK::pfnGetType */
285static DECLCALLBACK(PDMBLOCKTYPE) drvblockGetType(PPDMIBLOCK pInterface)
286{
287 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
288 LogFlow(("drvblockGetType: returns %d\n", pThis->enmType));
289 return pThis->enmType;
290}
291
292
293/** @copydoc PDMIBLOCK::pfnGetUuid */
294static DECLCALLBACK(int) drvblockGetUuid(PPDMIBLOCK pInterface, PRTUUID pUuid)
295{
296 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
297
298 /*
299 * Copy the uuid.
300 */
301 *pUuid = pThis->Uuid;
302 return VINF_SUCCESS;
303}
304
305static DECLCALLBACK(int) drvblockDiscard(PPDMIBLOCK pInterface, PCRTRANGE paRanges, unsigned cRanges)
306{
307 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
308
309 return pThis->pDrvMedia->pfnDiscard(pThis->pDrvMedia, paRanges, cRanges);
310}
311
312/** @copydoc PDMIBLOCK::pfnIoBufAlloc */
313static DECLCALLBACK(int) drvblockIoBufAlloc(PPDMIBLOCK pInterface, size_t cb, void **ppvNew)
314{
315 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
316
317 return pThis->pDrvMedia->pfnIoBufAlloc(pThis->pDrvMedia, cb, ppvNew);
318}
319
320/** @copydoc PDMIBLOCK::pfnIoBufFree */
321static DECLCALLBACK(int) drvblockIoBufFree(PPDMIBLOCK pInterface, void *pv, size_t cb)
322{
323 PDRVBLOCK pThis = PDMIBLOCK_2_DRVBLOCK(pInterface);
324
325 return pThis->pDrvMedia->pfnIoBufFree(pThis->pDrvMedia, pv, cb);
326}
327
328/* -=-=-=-=- IBlockAsync -=-=-=-=- */
329
330/** Makes a PDRVBLOCK out of a PPDMIBLOCKASYNC. */
331#define PDMIBLOCKASYNC_2_DRVBLOCK(pInterface) ( (PDRVBLOCK)((uintptr_t)pInterface - RT_OFFSETOF(DRVBLOCK, IBlockAsync)) )
332
333/** @copydoc PDMIBLOCKASYNC::pfnStartRead */
334static DECLCALLBACK(int) drvblockAsyncReadStart(PPDMIBLOCKASYNC pInterface, uint64_t off, PCRTSGSEG pSeg, unsigned cSeg, size_t cbRead, void *pvUser)
335{
336 PDRVBLOCK pThis = PDMIBLOCKASYNC_2_DRVBLOCK(pInterface);
337
338 /*
339 * Check the state.
340 */
341 if (!pThis->pDrvMediaAsync)
342 {
343 AssertMsgFailed(("Invalid state! Not mounted!\n"));
344 return VERR_PDM_MEDIA_NOT_MOUNTED;
345 }
346
347 int rc = pThis->pDrvMediaAsync->pfnStartRead(pThis->pDrvMediaAsync, off, pSeg, cSeg, cbRead, pvUser);
348 return rc;
349}
350
351
352/** @copydoc PDMIBLOCKASYNC::pfnStartWrite */
353static DECLCALLBACK(int) drvblockAsyncWriteStart(PPDMIBLOCKASYNC pInterface, uint64_t off, PCRTSGSEG pSeg, unsigned cSeg, size_t cbWrite, void *pvUser)
354{
355 PDRVBLOCK pThis = PDMIBLOCKASYNC_2_DRVBLOCK(pInterface);
356
357 /*
358 * Check the state.
359 */
360 if (!pThis->pDrvMediaAsync)
361 {
362 AssertMsgFailed(("Invalid state! Not mounted!\n"));
363 return VERR_PDM_MEDIA_NOT_MOUNTED;
364 }
365
366 int rc = pThis->pDrvMediaAsync->pfnStartWrite(pThis->pDrvMediaAsync, off, pSeg, cSeg, cbWrite, pvUser);
367
368 return rc;
369}
370
371
372/** @copydoc PDMIBLOCKASYNC::pfnStartFlush */
373static DECLCALLBACK(int) drvblockAsyncFlushStart(PPDMIBLOCKASYNC pInterface, void *pvUser)
374{
375 PDRVBLOCK pThis = PDMIBLOCKASYNC_2_DRVBLOCK(pInterface);
376
377 /*
378 * Check the state.
379 */
380 if (!pThis->pDrvMediaAsync)
381 {
382 AssertMsgFailed(("Invalid state! Not mounted!\n"));
383 return VERR_PDM_MEDIA_NOT_MOUNTED;
384 }
385
386#ifdef VBOX_IGNORE_FLUSH
387 if (pThis->fIgnoreFlushAsync)
388 return VINF_VD_ASYNC_IO_FINISHED;
389#endif /* VBOX_IGNORE_FLUSH */
390
391 int rc = pThis->pDrvMediaAsync->pfnStartFlush(pThis->pDrvMediaAsync, pvUser);
392
393 return rc;
394}
395
396
397/** @copydoc PDMIBLOCKASYNC::pfnStartDiscard */
398static DECLCALLBACK(int) drvblockStartDiscard(PPDMIBLOCKASYNC pInterface, PCRTRANGE paRanges, unsigned cRanges, void *pvUser)
399{
400 PDRVBLOCK pThis = PDMIBLOCKASYNC_2_DRVBLOCK(pInterface);
401
402 /*
403 * Check the state.
404 */
405 if (!pThis->pDrvMediaAsync)
406 {
407 AssertMsgFailed(("Invalid state! Not mounted!\n"));
408 return VERR_PDM_MEDIA_NOT_MOUNTED;
409 }
410
411 return pThis->pDrvMediaAsync->pfnStartDiscard(pThis->pDrvMediaAsync, paRanges, cRanges, pvUser);
412}
413
414/* -=-=-=-=- IMediaAsyncPort -=-=-=-=- */
415
416/** Makes a PDRVBLOCKASYNC out of a PPDMIMEDIAASYNCPORT. */
417#define PDMIMEDIAASYNCPORT_2_DRVBLOCK(pInterface) ( (PDRVBLOCK((uintptr_t)pInterface - RT_OFFSETOF(DRVBLOCK, IMediaAsyncPort))) )
418
419static DECLCALLBACK(int) drvblockAsyncTransferCompleteNotify(PPDMIMEDIAASYNCPORT pInterface, void *pvUser, int rcReq)
420{
421 PDRVBLOCK pThis = PDMIMEDIAASYNCPORT_2_DRVBLOCK(pInterface);
422
423 return pThis->pDrvBlockAsyncPort->pfnTransferCompleteNotify(pThis->pDrvBlockAsyncPort, pvUser, rcReq);
424}
425
426/* -=-=-=-=- IBlockBios -=-=-=-=- */
427
428/** Makes a PDRVBLOCK out of a PPDMIBLOCKBIOS. */
429#define PDMIBLOCKBIOS_2_DRVBLOCK(pInterface) ( (PDRVBLOCK((uintptr_t)pInterface - RT_OFFSETOF(DRVBLOCK, IBlockBios))) )
430
431
432/** @copydoc PDMIBLOCKBIOS::pfnGetPCHSGeometry */
433static DECLCALLBACK(int) drvblockGetPCHSGeometry(PPDMIBLOCKBIOS pInterface, PPDMMEDIAGEOMETRY pPCHSGeometry)
434{
435 PDRVBLOCK pThis = PDMIBLOCKBIOS_2_DRVBLOCK(pInterface);
436
437 /*
438 * Check the state.
439 */
440 if (!pThis->pDrvMedia)
441 return VERR_PDM_MEDIA_NOT_MOUNTED;
442
443 /*
444 * Use configured/cached values if present.
445 */
446 if ( pThis->PCHSGeometry.cCylinders > 0
447 && pThis->PCHSGeometry.cHeads > 0
448 && pThis->PCHSGeometry.cSectors > 0)
449 {
450 *pPCHSGeometry = pThis->PCHSGeometry;
451 LogFlow(("%s: returns VINF_SUCCESS {%d,%d,%d}\n", __FUNCTION__, pThis->PCHSGeometry.cCylinders, pThis->PCHSGeometry.cHeads, pThis->PCHSGeometry.cSectors));
452 return VINF_SUCCESS;
453 }
454
455 /*
456 * Call media.
457 */
458 int rc = pThis->pDrvMedia->pfnBiosGetPCHSGeometry(pThis->pDrvMedia, &pThis->PCHSGeometry);
459
460 if (RT_SUCCESS(rc))
461 {
462 *pPCHSGeometry = pThis->PCHSGeometry;
463 LogFlow(("%s: returns %Rrc {%d,%d,%d}\n", __FUNCTION__, rc, pThis->PCHSGeometry.cCylinders, pThis->PCHSGeometry.cHeads, pThis->PCHSGeometry.cSectors));
464 }
465 else if (rc == VERR_NOT_IMPLEMENTED)
466 {
467 rc = VERR_PDM_GEOMETRY_NOT_SET;
468 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
469 }
470 return rc;
471}
472
473
474/** @copydoc PDMIBLOCKBIOS::pfnSetPCHSGeometry */
475static DECLCALLBACK(int) drvblockSetPCHSGeometry(PPDMIBLOCKBIOS pInterface, PCPDMMEDIAGEOMETRY pPCHSGeometry)
476{
477 LogFlow(("%s: cCylinders=%d cHeads=%d cSectors=%d\n", __FUNCTION__, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
478 PDRVBLOCK pThis = PDMIBLOCKBIOS_2_DRVBLOCK(pInterface);
479
480 /*
481 * Check the state.
482 */
483 if (!pThis->pDrvMedia)
484 {
485 AssertMsgFailed(("Invalid state! Not mounted!\n"));
486 return VERR_PDM_MEDIA_NOT_MOUNTED;
487 }
488
489 /*
490 * Call media. Ignore the not implemented return code.
491 */
492 int rc = pThis->pDrvMedia->pfnBiosSetPCHSGeometry(pThis->pDrvMedia, pPCHSGeometry);
493
494 if ( RT_SUCCESS(rc)
495 || rc == VERR_NOT_IMPLEMENTED)
496 {
497 pThis->PCHSGeometry = *pPCHSGeometry;
498 rc = VINF_SUCCESS;
499 }
500 return rc;
501}
502
503
504/** @copydoc PDMIBLOCKBIOS::pfnGetLCHSGeometry */
505static DECLCALLBACK(int) drvblockGetLCHSGeometry(PPDMIBLOCKBIOS pInterface, PPDMMEDIAGEOMETRY pLCHSGeometry)
506{
507 PDRVBLOCK pThis = PDMIBLOCKBIOS_2_DRVBLOCK(pInterface);
508
509 /*
510 * Check the state.
511 */
512 if (!pThis->pDrvMedia)
513 return VERR_PDM_MEDIA_NOT_MOUNTED;
514
515 /*
516 * Use configured/cached values if present.
517 */
518 if ( pThis->LCHSGeometry.cCylinders > 0
519 && pThis->LCHSGeometry.cHeads > 0
520 && pThis->LCHSGeometry.cSectors > 0)
521 {
522 *pLCHSGeometry = pThis->LCHSGeometry;
523 LogFlow(("%s: returns VINF_SUCCESS {%d,%d,%d}\n", __FUNCTION__, pThis->LCHSGeometry.cCylinders, pThis->LCHSGeometry.cHeads, pThis->LCHSGeometry.cSectors));
524 return VINF_SUCCESS;
525 }
526
527 /*
528 * Call media.
529 */
530 int rc = pThis->pDrvMedia->pfnBiosGetLCHSGeometry(pThis->pDrvMedia, &pThis->LCHSGeometry);
531
532 if (RT_SUCCESS(rc))
533 {
534 *pLCHSGeometry = pThis->LCHSGeometry;
535 LogFlow(("%s: returns %Rrc {%d,%d,%d}\n", __FUNCTION__, rc, pThis->LCHSGeometry.cCylinders, pThis->LCHSGeometry.cHeads, pThis->LCHSGeometry.cSectors));
536 }
537 else if (rc == VERR_NOT_IMPLEMENTED)
538 {
539 rc = VERR_PDM_GEOMETRY_NOT_SET;
540 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
541 }
542 return rc;
543}
544
545
546/** @copydoc PDMIBLOCKBIOS::pfnSetLCHSGeometry */
547static DECLCALLBACK(int) drvblockSetLCHSGeometry(PPDMIBLOCKBIOS pInterface, PCPDMMEDIAGEOMETRY pLCHSGeometry)
548{
549 LogFlow(("%s: cCylinders=%d cHeads=%d cSectors=%d\n", __FUNCTION__, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
550 PDRVBLOCK pThis = PDMIBLOCKBIOS_2_DRVBLOCK(pInterface);
551
552 /*
553 * Check the state.
554 */
555 if (!pThis->pDrvMedia)
556 {
557 AssertMsgFailed(("Invalid state! Not mounted!\n"));
558 return VERR_PDM_MEDIA_NOT_MOUNTED;
559 }
560
561 /*
562 * Call media. Ignore the not implemented return code.
563 */
564 int rc = pThis->pDrvMedia->pfnBiosSetLCHSGeometry(pThis->pDrvMedia, pLCHSGeometry);
565
566 if ( RT_SUCCESS(rc)
567 || rc == VERR_NOT_IMPLEMENTED)
568 {
569 pThis->LCHSGeometry = *pLCHSGeometry;
570 rc = VINF_SUCCESS;
571 }
572 return rc;
573}
574
575
576/** @copydoc PDMIBLOCKBIOS::pfnIsVisible */
577static DECLCALLBACK(bool) drvblockIsVisible(PPDMIBLOCKBIOS pInterface)
578{
579 PDRVBLOCK pThis = PDMIBLOCKBIOS_2_DRVBLOCK(pInterface);
580 LogFlow(("drvblockIsVisible: returns %d\n", pThis->fBiosVisible));
581 return pThis->fBiosVisible;
582}
583
584
585/** @copydoc PDMIBLOCKBIOS::pfnGetType */
586static DECLCALLBACK(PDMBLOCKTYPE) drvblockBiosGetType(PPDMIBLOCKBIOS pInterface)
587{
588 PDRVBLOCK pThis = PDMIBLOCKBIOS_2_DRVBLOCK(pInterface);
589 LogFlow(("drvblockBiosGetType: returns %d\n", pThis->enmType));
590 return pThis->enmType;
591}
592
593
594
595/* -=-=-=-=- IMount -=-=-=-=- */
596
597/** Makes a PDRVBLOCK out of a PPDMIMOUNT. */
598#define PDMIMOUNT_2_DRVBLOCK(pInterface) ( (PDRVBLOCK)((uintptr_t)pInterface - RT_OFFSETOF(DRVBLOCK, IMount)) )
599
600
601/** @copydoc PDMIMOUNT::pfnMount */
602static DECLCALLBACK(int) drvblockMount(PPDMIMOUNT pInterface, const char *pszFilename, const char *pszCoreDriver)
603{
604 LogFlow(("drvblockMount: pszFilename=%p:{%s} pszCoreDriver=%p:{%s}\n", pszFilename, pszFilename, pszCoreDriver, pszCoreDriver));
605 PDRVBLOCK pThis = PDMIMOUNT_2_DRVBLOCK(pInterface);
606
607 /*
608 * Validate state.
609 */
610 if (pThis->pDrvMedia)
611 {
612 AssertMsgFailed(("Already mounted\n"));
613 return VERR_PDM_MEDIA_MOUNTED;
614 }
615
616 /*
617 * Prepare configuration.
618 */
619 if (pszFilename)
620 {
621 int rc = PDMDrvHlpMountPrepare(pThis->pDrvIns, pszFilename, pszCoreDriver);
622 if (RT_FAILURE(rc))
623 {
624 Log(("drvblockMount: Prepare failed for \"%s\" rc=%Rrc\n", pszFilename, rc));
625 return rc;
626 }
627 }
628
629 /*
630 * Attach the media driver and query it's interface.
631 */
632 uint32_t fTachFlags = 0; /** @todo figure attachment flags for mount. */
633 PPDMIBASE pBase;
634 int rc = PDMDrvHlpAttach(pThis->pDrvIns, fTachFlags, &pBase);
635 if (RT_FAILURE(rc))
636 {
637 Log(("drvblockMount: Attach failed rc=%Rrc\n", rc));
638 return rc;
639 }
640
641 pThis->pDrvMedia = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMEDIA);
642 if (pThis->pDrvMedia)
643 {
644 /** @todo r=klaus missing async handling, this is just a band aid to
645 * avoid using stale information */
646 pThis->pDrvMediaAsync = NULL;
647
648 /*
649 * Initialize state.
650 */
651 pThis->fLocked = false;
652 pThis->PCHSGeometry.cCylinders = 0;
653 pThis->PCHSGeometry.cHeads = 0;
654 pThis->PCHSGeometry.cSectors = 0;
655 pThis->LCHSGeometry.cCylinders = 0;
656 pThis->LCHSGeometry.cHeads = 0;
657 pThis->LCHSGeometry.cSectors = 0;
658#ifdef VBOX_PERIODIC_FLUSH
659 pThis->cbDataWritten = 0;
660#endif /* VBOX_PERIODIC_FLUSH */
661
662 /*
663 * Notify driver/device above us.
664 */
665 if (pThis->pDrvMountNotify)
666 pThis->pDrvMountNotify->pfnMountNotify(pThis->pDrvMountNotify);
667 Log(("drvblockMount: Success\n"));
668 return VINF_SUCCESS;
669 }
670 else
671 rc = VERR_PDM_MISSING_INTERFACE_BELOW;
672
673 /*
674 * Failed, detatch the media driver.
675 */
676 AssertMsgFailed(("No media interface!\n"));
677 int rc2 = PDMDrvHlpDetach(pThis->pDrvIns, fTachFlags);
678 AssertRC(rc2);
679 pThis->pDrvMedia = NULL;
680 return rc;
681}
682
683
684/** @copydoc PDMIMOUNT::pfnUnmount */
685static DECLCALLBACK(int) drvblockUnmount(PPDMIMOUNT pInterface, bool fForce, bool fEject)
686{
687 PDRVBLOCK pThis = PDMIMOUNT_2_DRVBLOCK(pInterface);
688
689 /*
690 * Validate state.
691 */
692 if (!pThis->pDrvMedia)
693 {
694 Log(("drvblockUmount: Not mounted\n"));
695 return VERR_PDM_MEDIA_NOT_MOUNTED;
696 }
697 if (pThis->fLocked && !fForce)
698 {
699 Log(("drvblockUmount: Locked\n"));
700 return VERR_PDM_MEDIA_LOCKED;
701 }
702
703 /* Media is no longer locked even if it was previously. */
704 pThis->fLocked = false;
705
706 /*
707 * Detach the media driver and query it's interface.
708 */
709 int rc = PDMDrvHlpDetach(pThis->pDrvIns, 0 /*fFlags*/);
710 if (RT_FAILURE(rc))
711 {
712 Log(("drvblockUnmount: Detach failed rc=%Rrc\n", rc));
713 return rc;
714 }
715 Assert(!pThis->pDrvMedia);
716
717 /*
718 * Notify driver/device above us.
719 */
720 if (pThis->pDrvMountNotify)
721 pThis->pDrvMountNotify->pfnUnmountNotify(pThis->pDrvMountNotify);
722 Log(("drvblockUnmount: success\n"));
723 return VINF_SUCCESS;
724}
725
726
727/** @copydoc PDMIMOUNT::pfnIsMounted */
728static DECLCALLBACK(bool) drvblockIsMounted(PPDMIMOUNT pInterface)
729{
730 PDRVBLOCK pThis = PDMIMOUNT_2_DRVBLOCK(pInterface);
731 return pThis->pDrvMedia != NULL;
732}
733
734/** @copydoc PDMIMOUNT::pfnLock */
735static DECLCALLBACK(int) drvblockLock(PPDMIMOUNT pInterface)
736{
737 PDRVBLOCK pThis = PDMIMOUNT_2_DRVBLOCK(pInterface);
738 Log(("drvblockLock: %d -> %d\n", pThis->fLocked, true));
739 pThis->fLocked = true;
740 return VINF_SUCCESS;
741}
742
743/** @copydoc PDMIMOUNT::pfnUnlock */
744static DECLCALLBACK(int) drvblockUnlock(PPDMIMOUNT pInterface)
745{
746 PDRVBLOCK pThis = PDMIMOUNT_2_DRVBLOCK(pInterface);
747 Log(("drvblockUnlock: %d -> %d\n", pThis->fLocked, false));
748 pThis->fLocked = false;
749 return VINF_SUCCESS;
750}
751
752/** @copydoc PDMIMOUNT::pfnIsLocked */
753static DECLCALLBACK(bool) drvblockIsLocked(PPDMIMOUNT pInterface)
754{
755 PDRVBLOCK pThis = PDMIMOUNT_2_DRVBLOCK(pInterface);
756 return pThis->fLocked;
757}
758
759
760
761/* -=-=-=-=- IMediaPort -=-=-=-=- */
762
763/** Makes a PDRVBLOCK out of a PPDMIMEDIAPORT. */
764#define PDMIMEDIAPORT_2_DRVBLOCK(pInterface) ( (PDRVBLOCK((uintptr_t)pInterface - RT_OFFSETOF(DRVBLOCK, IMediaPort))) )
765
766/**
767 * @interface_method_impl{PDMIMEDIAPORT,pfnQueryDeviceLocation}
768 */
769static DECLCALLBACK(int) drvblockQueryDeviceLocation(PPDMIMEDIAPORT pInterface, const char **ppcszController,
770 uint32_t *piInstance, uint32_t *piLUN)
771{
772 PDRVBLOCK pThis = PDMIMEDIAPORT_2_DRVBLOCK(pInterface);
773
774 return pThis->pDrvBlockPort->pfnQueryDeviceLocation(pThis->pDrvBlockPort, ppcszController,
775 piInstance, piLUN);
776}
777
778/* -=-=-=-=- IBase -=-=-=-=- */
779
780/**
781 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
782 */
783static DECLCALLBACK(void *) drvblockQueryInterface(PPDMIBASE pInterface, const char *pszIID)
784{
785 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
786 PDRVBLOCK pThis = PDMINS_2_DATA(pDrvIns, PDRVBLOCK);
787
788 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
789 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBLOCK, &pThis->IBlock);
790 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBLOCKBIOS, pThis->fBiosVisible ? &pThis->IBlockBios : NULL);
791 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMOUNT, pThis->fMountable ? &pThis->IMount : NULL);
792 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBLOCKASYNC, pThis->pDrvMediaAsync ? &pThis->IBlockAsync : NULL);
793 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAASYNCPORT, pThis->pDrvBlockAsyncPort ? &pThis->IMediaAsyncPort : NULL);
794 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAPORT, &pThis->IMediaPort);
795 return NULL;
796}
797
798
799/* -=-=-=-=- driver interface -=-=-=-=- */
800
801/** @copydoc FNPDMDRVDETACH. */
802static DECLCALLBACK(void) drvblockDetach(PPDMDRVINS pDrvIns, uint32_t fFlags)
803{
804 PDRVBLOCK pThis = PDMINS_2_DATA(pDrvIns, PDRVBLOCK);
805 pThis->pDrvMedia = NULL;
806 pThis->pDrvMediaAsync = NULL;
807 NOREF(fFlags);
808}
809
810
811/**
812 * Reset notification.
813 *
814 * @returns VBox status.
815 * @param pDevIns The driver instance data.
816 */
817static DECLCALLBACK(void) drvblockReset(PPDMDRVINS pDrvIns)
818{
819 PDRVBLOCK pThis = PDMINS_2_DATA(pDrvIns, PDRVBLOCK);
820
821 pThis->fLocked = false;
822}
823
824
825/**
826 * Translates a PDMBLOCKTYPE value into a string.
827 *
828 * @returns Read only string.
829 * @param enmType The type value.
830 */
831static const char *drvblockGetTypeName(PDMBLOCKTYPE enmType)
832{
833 switch (enmType)
834 {
835 case PDMBLOCKTYPE_ERROR: return "ERROR";
836 case PDMBLOCKTYPE_FLOPPY_360: return "FLOPPY_360";
837 case PDMBLOCKTYPE_FLOPPY_720: return "FLOPPY_720";
838 case PDMBLOCKTYPE_FLOPPY_1_20: return "FLOPPY_1_20";
839 case PDMBLOCKTYPE_FLOPPY_1_44: return "FLOPPY_1_44";
840 case PDMBLOCKTYPE_FLOPPY_2_88: return "FLOPPY_2_88";
841 case PDMBLOCKTYPE_FLOPPY_FAKE_15_6: return "FLOPPY_FAKE_15_6";
842 case PDMBLOCKTYPE_FLOPPY_FAKE_63_5: return "FLOPPY_FAKE_63_5";
843 case PDMBLOCKTYPE_CDROM: return "CDROM";
844 case PDMBLOCKTYPE_DVD: return "DVD";
845 case PDMBLOCKTYPE_HARD_DISK: return "HARD_DISK";
846 default: return "Unknown";
847
848 }
849}
850
851
852/**
853 * Construct a block driver instance.
854 *
855 * @copydoc FNPDMDRVCONSTRUCT
856 */
857static DECLCALLBACK(int) drvblockConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
858{
859 PDRVBLOCK pThis = PDMINS_2_DATA(pDrvIns, PDRVBLOCK);
860 LogFlow(("drvblockConstruct: iInstance=%d\n", pDrvIns->iInstance));
861 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
862
863 /*
864 * Validate configuration.
865 */
866#if defined(VBOX_PERIODIC_FLUSH) || defined(VBOX_IGNORE_FLUSH)
867 if (!CFGMR3AreValuesValid(pCfg, "Type\0Locked\0BIOSVisible\0AttachFailError\0Cylinders\0Heads\0Sectors\0Mountable\0FlushInterval\0IgnoreFlush\0IgnoreFlushAsync\0"))
868#else /* !(VBOX_PERIODIC_FLUSH || VBOX_IGNORE_FLUSH) */
869 if (!CFGMR3AreValuesValid(pCfg, "Type\0Locked\0BIOSVisible\0AttachFailError\0Cylinders\0Heads\0Sectors\0Mountable\0"))
870#endif /* !(VBOX_PERIODIC_FLUSH || VBOX_IGNORE_FLUSH) */
871 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
872
873 /*
874 * Initialize most of the data members.
875 */
876 pThis->pDrvIns = pDrvIns;
877
878 /* IBase. */
879 pDrvIns->IBase.pfnQueryInterface = drvblockQueryInterface;
880
881 /* IBlock. */
882 pThis->IBlock.pfnRead = drvblockRead;
883 pThis->IBlock.pfnWrite = drvblockWrite;
884 pThis->IBlock.pfnFlush = drvblockFlush;
885 pThis->IBlock.pfnMerge = drvblockMerge;
886 pThis->IBlock.pfnIsReadOnly = drvblockIsReadOnly;
887 pThis->IBlock.pfnGetSize = drvblockGetSize;
888 pThis->IBlock.pfnGetSectorSize = drvblockGetSectorSize;
889 pThis->IBlock.pfnGetType = drvblockGetType;
890 pThis->IBlock.pfnGetUuid = drvblockGetUuid;
891 pThis->IBlock.pfnIoBufAlloc = drvblockIoBufAlloc;
892 pThis->IBlock.pfnIoBufFree = drvblockIoBufFree;
893
894 /* IBlockBios. */
895 pThis->IBlockBios.pfnGetPCHSGeometry = drvblockGetPCHSGeometry;
896 pThis->IBlockBios.pfnSetPCHSGeometry = drvblockSetPCHSGeometry;
897 pThis->IBlockBios.pfnGetLCHSGeometry = drvblockGetLCHSGeometry;
898 pThis->IBlockBios.pfnSetLCHSGeometry = drvblockSetLCHSGeometry;
899 pThis->IBlockBios.pfnIsVisible = drvblockIsVisible;
900 pThis->IBlockBios.pfnGetType = drvblockBiosGetType;
901
902 /* IMount. */
903 pThis->IMount.pfnMount = drvblockMount;
904 pThis->IMount.pfnUnmount = drvblockUnmount;
905 pThis->IMount.pfnIsMounted = drvblockIsMounted;
906 pThis->IMount.pfnLock = drvblockLock;
907 pThis->IMount.pfnUnlock = drvblockUnlock;
908 pThis->IMount.pfnIsLocked = drvblockIsLocked;
909
910 /* IBlockAsync. */
911 pThis->IBlockAsync.pfnStartRead = drvblockAsyncReadStart;
912 pThis->IBlockAsync.pfnStartWrite = drvblockAsyncWriteStart;
913 pThis->IBlockAsync.pfnStartFlush = drvblockAsyncFlushStart;
914
915 /* IMediaAsyncPort. */
916 pThis->IMediaAsyncPort.pfnTransferCompleteNotify = drvblockAsyncTransferCompleteNotify;
917
918 /* IMediaPort */
919 pThis->IMediaPort.pfnQueryDeviceLocation = drvblockQueryDeviceLocation;
920
921 /*
922 * Get the IBlockPort & IMountNotify interfaces of the above driver/device.
923 */
924 pThis->pDrvBlockPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIBLOCKPORT);
925 if (!pThis->pDrvBlockPort)
926 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_MISSING_INTERFACE_ABOVE,
927 N_("No block port interface above"));
928
929 /* Try to get the optional async block port interface above. */
930 pThis->pDrvBlockAsyncPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIBLOCKASYNCPORT);
931 pThis->pDrvMountNotify = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIMOUNTNOTIFY);
932
933 /*
934 * Query configuration.
935 */
936 /* type */
937 char *psz;
938 int rc = CFGMR3QueryStringAlloc(pCfg, "Type", &psz);
939 if (RT_FAILURE(rc))
940 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_BLOCK_NO_TYPE, N_("Failed to obtain the type"));
941 if (!strcmp(psz, "HardDisk"))
942 pThis->enmType = PDMBLOCKTYPE_HARD_DISK;
943 else if (!strcmp(psz, "DVD"))
944 pThis->enmType = PDMBLOCKTYPE_DVD;
945 else if (!strcmp(psz, "CDROM"))
946 pThis->enmType = PDMBLOCKTYPE_CDROM;
947 else if (!strcmp(psz, "Floppy 2.88"))
948 pThis->enmType = PDMBLOCKTYPE_FLOPPY_2_88;
949 else if (!strcmp(psz, "Floppy 1.44"))
950 pThis->enmType = PDMBLOCKTYPE_FLOPPY_1_44;
951 else if (!strcmp(psz, "Floppy 1.20"))
952 pThis->enmType = PDMBLOCKTYPE_FLOPPY_1_20;
953 else if (!strcmp(psz, "Floppy 720"))
954 pThis->enmType = PDMBLOCKTYPE_FLOPPY_720;
955 else if (!strcmp(psz, "Floppy 360"))
956 pThis->enmType = PDMBLOCKTYPE_FLOPPY_360;
957 else if (!strcmp(psz, "Floppy 15.6"))
958 pThis->enmType = PDMBLOCKTYPE_FLOPPY_FAKE_15_6;
959 else if (!strcmp(psz, "Floppy 63.5"))
960 pThis->enmType = PDMBLOCKTYPE_FLOPPY_FAKE_63_5;
961 else
962 {
963 PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_BLOCK_UNKNOWN_TYPE, RT_SRC_POS,
964 N_("Unknown type \"%s\""), psz);
965 MMR3HeapFree(psz);
966 return VERR_PDM_BLOCK_UNKNOWN_TYPE;
967 }
968 Log2(("drvblockConstruct: enmType=%d\n", pThis->enmType));
969 MMR3HeapFree(psz); psz = NULL;
970
971 /* Mountable */
972 rc = CFGMR3QueryBoolDef(pCfg, "Mountable", &pThis->fMountable, false);
973 if (RT_FAILURE(rc))
974 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"Mountable\" from the config"));
975
976 /* Locked */
977 rc = CFGMR3QueryBoolDef(pCfg, "Locked", &pThis->fLocked, false);
978 if (RT_FAILURE(rc))
979 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"Locked\" from the config"));
980
981 /* BIOS visible */
982 rc = CFGMR3QueryBoolDef(pCfg, "BIOSVisible", &pThis->fBiosVisible, true);
983 if (RT_FAILURE(rc))
984 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"BIOSVisible\" from the config"));
985
986 /** @todo AttachFailError is currently completely ignored. */
987
988 /* Cylinders */
989 rc = CFGMR3QueryU32Def(pCfg, "Cylinders", &pThis->LCHSGeometry.cCylinders, 0);
990 if (RT_FAILURE(rc))
991 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"Cylinders\" from the config"));
992
993 /* Heads */
994 rc = CFGMR3QueryU32Def(pCfg, "Heads", &pThis->LCHSGeometry.cHeads, 0);
995 if (RT_FAILURE(rc))
996 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"Heads\" from the config"));
997
998 /* Sectors */
999 rc = CFGMR3QueryU32Def(pCfg, "Sectors", &pThis->LCHSGeometry.cSectors, 0);
1000 if (RT_FAILURE(rc))
1001 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"Sectors\" from the config"));
1002
1003 /* Uuid */
1004 rc = CFGMR3QueryStringAlloc(pCfg, "Uuid", &psz);
1005 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1006 RTUuidClear(&pThis->Uuid);
1007 else if (RT_SUCCESS(rc))
1008 {
1009 rc = RTUuidFromStr(&pThis->Uuid, psz);
1010 if (RT_FAILURE(rc))
1011 {
1012 PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS, "%s",
1013 N_("Uuid from string failed on \"%s\""), psz);
1014 MMR3HeapFree(psz);
1015 return rc;
1016 }
1017 MMR3HeapFree(psz); psz = NULL;
1018 }
1019 else
1020 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"Uuid\" from the config"));
1021
1022#ifdef VBOX_PERIODIC_FLUSH
1023 rc = CFGMR3QueryU32Def(pCfg, "FlushInterval", &pThis->cbFlushInterval, 0);
1024 if (RT_FAILURE(rc))
1025 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"FlushInterval\" from the config"));
1026#endif /* VBOX_PERIODIC_FLUSH */
1027
1028#ifdef VBOX_IGNORE_FLUSH
1029 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreFlush", &pThis->fIgnoreFlush, true);
1030 if (RT_FAILURE(rc))
1031 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"IgnoreFlush\" from the config"));
1032
1033 if (pThis->fIgnoreFlush)
1034 LogRel(("DrvBlock: Flushes will be ignored\n"));
1035 else
1036 LogRel(("DrvBlock: Flushes will be passed to the disk\n"));
1037
1038 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreFlushAsync", &pThis->fIgnoreFlushAsync, false);
1039 if (RT_FAILURE(rc))
1040 return PDMDRV_SET_ERROR(pDrvIns, rc, N_("Failed to query \"IgnoreFlushAsync\" from the config"));
1041
1042 if (pThis->fIgnoreFlushAsync)
1043 LogRel(("DrvBlock: Async flushes will be ignored\n"));
1044 else
1045 LogRel(("DrvBlock: Async flushes will be passed to the disk\n"));
1046#endif /* VBOX_IGNORE_FLUSH */
1047
1048 /*
1049 * Try attach driver below and query it's media interface.
1050 */
1051 PPDMIBASE pBase;
1052 rc = PDMDrvHlpAttach(pDrvIns, fFlags, &pBase);
1053 if ( rc == VERR_PDM_NO_ATTACHED_DRIVER
1054 && pThis->enmType != PDMBLOCKTYPE_HARD_DISK)
1055 return VINF_SUCCESS;
1056 if (RT_FAILURE(rc))
1057 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1058 N_("Failed to attach driver below us! %Rrf"), rc);
1059
1060 pThis->pDrvMedia = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMEDIA);
1061 if (!pThis->pDrvMedia)
1062 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_MISSING_INTERFACE_BELOW,
1063 N_("No media or async media interface below"));
1064
1065 /* Try to get the optional async interface. */
1066 pThis->pDrvMediaAsync = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMEDIAASYNC);
1067
1068 if (pThis->pDrvMedia->pfnDiscard)
1069 pThis->IBlock.pfnDiscard = drvblockDiscard;
1070
1071 if ( pThis->pDrvMediaAsync
1072 && pThis->pDrvMediaAsync->pfnStartDiscard)
1073 pThis->IBlockAsync.pfnStartDiscard = drvblockStartDiscard;
1074
1075 if (RTUuidIsNull(&pThis->Uuid))
1076 {
1077 if (pThis->enmType == PDMBLOCKTYPE_HARD_DISK)
1078 pThis->pDrvMedia->pfnGetUuid(pThis->pDrvMedia, &pThis->Uuid);
1079 }
1080
1081 /*
1082 * Automatically upgrade the floppy drive if the specified one is too
1083 * small to represent the whole boot time image. (We cannot do this later
1084 * since the BIOS (and others) gets the info via CMOS.)
1085 *
1086 * This trick should make 2.88 images as well as the fake 15.6 and 63.5 MB
1087 * images despite the hardcoded default 1.44 drive.
1088 */
1089 if ( PDMBLOCKTYPE_IS_FLOPPY(pThis->enmType)
1090 && pThis->pDrvMedia)
1091 {
1092 uint64_t const cbFloppyImg = pThis->pDrvMedia->pfnGetSize(pThis->pDrvMedia);
1093 PDMBLOCKTYPE const enmCfgType = pThis->enmType;
1094 switch (enmCfgType)
1095 {
1096 default:
1097 AssertFailed();
1098 case PDMBLOCKTYPE_FLOPPY_360:
1099 if (cbFloppyImg > 40 * 2 * 9 * 512)
1100 pThis->enmType = PDMBLOCKTYPE_FLOPPY_360;
1101 /* fall thru */
1102 case PDMBLOCKTYPE_FLOPPY_720:
1103 if (cbFloppyImg > 80 * 2 * 14 * 512)
1104 pThis->enmType = PDMBLOCKTYPE_FLOPPY_1_20;
1105 /* fall thru */
1106 case PDMBLOCKTYPE_FLOPPY_1_20:
1107 if (cbFloppyImg > 80 * 2 * 20 * 512)
1108 pThis->enmType = PDMBLOCKTYPE_FLOPPY_1_44;
1109 /* fall thru */
1110 case PDMBLOCKTYPE_FLOPPY_1_44:
1111 if (cbFloppyImg > 80 * 2 * 24 * 512)
1112 pThis->enmType = PDMBLOCKTYPE_FLOPPY_2_88;
1113 /* fall thru */
1114 case PDMBLOCKTYPE_FLOPPY_2_88:
1115 if (cbFloppyImg > 80 * 2 * 48 * 512)
1116 pThis->enmType = PDMBLOCKTYPE_FLOPPY_FAKE_15_6;
1117 /* fall thru */
1118 case PDMBLOCKTYPE_FLOPPY_FAKE_15_6:
1119 if (cbFloppyImg > 255 * 2 * 63 * 512)
1120 pThis->enmType = PDMBLOCKTYPE_FLOPPY_FAKE_63_5;
1121 case PDMBLOCKTYPE_FLOPPY_FAKE_63_5:
1122 if (cbFloppyImg > 255 * 2 * 255 * 512)
1123 LogRel(("Warning: Floppy image is larger that 63.5 MB! (%llu bytes)\n", cbFloppyImg));
1124 break;
1125 }
1126 if (pThis->enmType != enmCfgType)
1127 LogRel(("Automatically upgraded floppy drive from %s to %s to better support the %u byte image\n",
1128 drvblockGetTypeName(enmCfgType), drvblockGetTypeName(pThis->enmType), cbFloppyImg));
1129 }
1130
1131 return VINF_SUCCESS;
1132}
1133
1134
1135/**
1136 * Block driver registration record.
1137 */
1138const PDMDRVREG g_DrvBlock =
1139{
1140 /* u32Version */
1141 PDM_DRVREG_VERSION,
1142 /* szName */
1143 "Block",
1144 /* szRCMod */
1145 "",
1146 /* szR0Mod */
1147 "",
1148 /* pszDescription */
1149 "Generic block driver.",
1150 /* fFlags */
1151 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1152 /* fClass. */
1153 PDM_DRVREG_CLASS_BLOCK,
1154 /* cMaxInstances */
1155 ~0U,
1156 /* cbInstance */
1157 sizeof(DRVBLOCK),
1158 /* pfnConstruct */
1159 drvblockConstruct,
1160 /* pfnDestruct */
1161 NULL,
1162 /* pfnRelocate */
1163 NULL,
1164 /* pfnIOCtl */
1165 NULL,
1166 /* pfnPowerOn */
1167 NULL,
1168 /* pfnReset */
1169 drvblockReset,
1170 /* pfnSuspend */
1171 NULL,
1172 /* pfnResume */
1173 NULL,
1174 /* pfnAttach */
1175 NULL,
1176 /* pfnDetach */
1177 drvblockDetach,
1178 /* pfnPowerOff */
1179 NULL,
1180 /* pfnSoftReset */
1181 NULL,
1182 /* u32EndVersion */
1183 PDM_DRVREG_VERSION
1184};
1185
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