VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvDiskIntegrity.cpp@ 59252

Last change on this file since 59252 was 59252, checked in by vboxsync, 9 years ago

pdmifs.h: Move the storage related interfaces (PDMIMEDIA, PDMIMOUNT, PDMISCSICONNECTOR, etc.) into a separate header to reduce the overall size of the header a bit

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 53.9 KB
Line 
1/* $Id: DrvDiskIntegrity.cpp 59252 2016-01-05 10:54:49Z vboxsync $ */
2/** @file
3 * VBox storage devices: Disk integrity check.
4 */
5
6/*
7 * Copyright (C) 2006-2015 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_DISK_INTEGRITY
23#include <VBox/vmm/pdmdrv.h>
24#include <VBox/vmm/pdmstorageifs.h>
25#include <VBox/vddbg.h>
26#include <iprt/assert.h>
27#include <iprt/string.h>
28#include <iprt/uuid.h>
29#include <iprt/avl.h>
30#include <iprt/mem.h>
31#include <iprt/message.h>
32#include <iprt/sg.h>
33#include <iprt/time.h>
34#include <iprt/semaphore.h>
35#include <iprt/asm.h>
36
37#include "VBoxDD.h"
38
39
40/*********************************************************************************************************************************
41* Structures and Typedefs *
42*********************************************************************************************************************************/
43
44/**
45 * Transfer direction.
46 */
47typedef enum DRVDISKAIOTXDIR
48{
49 /** Read */
50 DRVDISKAIOTXDIR_READ = 0,
51 /** Write */
52 DRVDISKAIOTXDIR_WRITE,
53 /** Flush */
54 DRVDISKAIOTXDIR_FLUSH,
55 /** Discard */
56 DRVDISKAIOTXDIR_DISCARD,
57 /** Read after write for immediate verification. */
58 DRVDISKAIOTXDIR_READ_AFTER_WRITE
59} DRVDISKAIOTXDIR;
60
61/**
62 * async I/O request.
63 */
64typedef struct DRVDISKAIOREQ
65{
66 /** Transfer direction. */
67 DRVDISKAIOTXDIR enmTxDir;
68 /** Start offset. */
69 uint64_t off;
70 /** Transfer size. */
71 size_t cbTransfer;
72 /** Segment array. */
73 PCRTSGSEG paSeg;
74 /** Number of array entries. */
75 unsigned cSeg;
76 /** User argument */
77 void *pvUser;
78 /** Slot in the array. */
79 unsigned iSlot;
80 /** Start timestamp */
81 uint64_t tsStart;
82 /** Completion timestamp. */
83 uint64_t tsComplete;
84 /** I/O log entry if configured. */
85 VDIOLOGENT hIoLogEntry;
86 /** Ranges to discard. */
87 PCRTRANGE paRanges;
88 /** Number of ranges. */
89 unsigned cRanges;
90} DRVDISKAIOREQ, *PDRVDISKAIOREQ;
91
92/**
93 * I/O log entry.
94 */
95typedef struct IOLOGENT
96{
97 /** Start offset */
98 uint64_t off;
99 /** Write size */
100 size_t cbWrite;
101 /** Number of references to this entry. */
102 unsigned cRefs;
103} IOLOGENT, *PIOLOGENT;
104
105/**
106 * Disk segment.
107 */
108typedef struct DRVDISKSEGMENT
109{
110 /** AVL core. */
111 AVLRFOFFNODECORE Core;
112 /** Size of the segment */
113 size_t cbSeg;
114 /** Data for this segment */
115 uint8_t *pbSeg;
116 /** Number of entries in the I/O array. */
117 unsigned cIoLogEntries;
118 /** Array of I/O log references. */
119 PIOLOGENT apIoLog[1];
120} DRVDISKSEGMENT, *PDRVDISKSEGMENT;
121
122/**
123 * Active requests list entry.
124 */
125typedef struct DRVDISKAIOREQACTIVE
126{
127 /** Pointer to the request. */
128 volatile PDRVDISKAIOREQ pIoReq;
129 /** Start timestamp. */
130 uint64_t tsStart;
131} DRVDISKAIOREQACTIVE, *PDRVDISKAIOREQACTIVE;
132
133/**
134 * Disk integrity driver instance data.
135 *
136 * @implements PDMIMEDIA
137 */
138typedef struct DRVDISKINTEGRITY
139{
140 /** Pointer driver instance. */
141 PPDMDRVINS pDrvIns;
142 /** Pointer to the media driver below us.
143 * This is NULL if the media is not mounted. */
144 PPDMIMEDIA pDrvMedia;
145 /** Our media interface */
146 PDMIMEDIA IMedia;
147
148 /** The media port interface above. */
149 PPDMIMEDIAPORT pDrvMediaPort;
150 /** Media port interface */
151 PDMIMEDIAPORT IMediaPort;
152
153 /** Pointer to the media async driver below us.
154 * This is NULL if the media is not mounted. */
155 PPDMIMEDIAASYNC pDrvMediaAsync;
156 /** Our media async interface */
157 PDMIMEDIAASYNC IMediaAsync;
158
159 /** The async media port interface above. */
160 PPDMIMEDIAASYNCPORT pDrvMediaAsyncPort;
161 /** Our media async port interface */
162 PDMIMEDIAASYNCPORT IMediaAsyncPort;
163
164 /** Flag whether consistency checks are enabled. */
165 bool fCheckConsistency;
166 /** Flag whether the RAM disk was prepopulated. */
167 bool fPrepopulateRamDisk;
168 /** AVL tree containing the disk blocks to check. */
169 PAVLRFOFFTREE pTreeSegments;
170
171 /** Flag whether async request tracing is enabled. */
172 bool fTraceRequests;
173 /** Interval the thread should check for expired requests (milliseconds). */
174 uint32_t uCheckIntervalMs;
175 /** Expire timeout for a request (milliseconds). */
176 uint32_t uExpireIntervalMs;
177 /** Thread which checks for lost requests. */
178 RTTHREAD hThread;
179 /** Event semaphore */
180 RTSEMEVENT SemEvent;
181 /** Flag whether the thread should run. */
182 bool fRunning;
183 /** Array containing active requests. */
184 DRVDISKAIOREQACTIVE apReqActive[128];
185 /** Next free slot in the array */
186 volatile unsigned iNextFreeSlot;
187
188 /** Flag whether we check for requests completing twice. */
189 bool fCheckDoubleCompletion;
190 /** Number of requests we go back. */
191 unsigned cEntries;
192 /** Array of completed but still observed requests. */
193 PDRVDISKAIOREQ *papIoReq;
194 /** Current entry in the array. */
195 unsigned iEntry;
196
197 /** Flag whether to do a immediate read after write for verification. */
198 bool fReadAfterWrite;
199 /** Flag whether to record the data to write before the write completed successfully.
200 * Useful in case the data is modified in place later on (encryption for instance). */
201 bool fRecordWriteBeforeCompletion;
202
203 /** I/O logger to use if enabled. */
204 VDIOLOGGER hIoLogger;
205} DRVDISKINTEGRITY, *PDRVDISKINTEGRITY;
206
207
208/**
209 * Allocate a new I/O request.
210 *
211 * @returns New I/O request.
212 * @param enmTxDir Transfer direction.
213 * @param off Start offset.
214 * @param paSeg Segment array.
215 * @param cSeg Number of segments.
216 * @param cbTransfer Number of bytes to transfer.
217 * @param pvUser User argument.
218 */
219static PDRVDISKAIOREQ drvdiskintIoReqAlloc(DRVDISKAIOTXDIR enmTxDir, uint64_t off, PCRTSGSEG paSeg,
220 unsigned cSeg, size_t cbTransfer, void *pvUser)
221{
222 PDRVDISKAIOREQ pIoReq = (PDRVDISKAIOREQ)RTMemAlloc(sizeof(DRVDISKAIOREQ));
223
224 if (RT_LIKELY(pIoReq))
225 {
226 pIoReq->enmTxDir = enmTxDir;
227 pIoReq->off = off;
228 pIoReq->cbTransfer = cbTransfer;
229 pIoReq->paSeg = paSeg;
230 pIoReq->cSeg = cSeg;
231 pIoReq->pvUser = pvUser;
232 pIoReq->iSlot = 0;
233 pIoReq->tsStart = RTTimeSystemMilliTS();
234 pIoReq->tsComplete = 0;
235 pIoReq->hIoLogEntry = NULL;
236 }
237
238 return pIoReq;
239}
240
241/**
242 * Free a async I/O request.
243 *
244 * @returns nothing.
245 * @param pThis Disk driver.
246 * @param pIoReq The I/O request to free.
247 */
248static void drvdiskintIoReqFree(PDRVDISKINTEGRITY pThis, PDRVDISKAIOREQ pIoReq)
249{
250 if (pThis->fCheckDoubleCompletion)
251 {
252 /* Search if the I/O request completed already. */
253 for (unsigned i = 0; i < pThis->cEntries; i++)
254 {
255 if (RT_UNLIKELY(pThis->papIoReq[i] == pIoReq))
256 {
257 RTMsgError("Request %#p completed already!\n", pIoReq);
258 RTMsgError("Start timestamp %llu Completion timestamp %llu (completed after %llu ms)\n",
259 pIoReq->tsStart, pIoReq->tsComplete, pIoReq->tsComplete - pIoReq->tsStart);
260 RTAssertDebugBreak();
261 }
262 }
263
264 pIoReq->tsComplete = RTTimeSystemMilliTS();
265 Assert(!pThis->papIoReq[pThis->iEntry]);
266 pThis->papIoReq[pThis->iEntry] = pIoReq;
267
268 pThis->iEntry = (pThis->iEntry+1) % pThis->cEntries;
269 if (pThis->papIoReq[pThis->iEntry])
270 {
271 RTMemFree(pThis->papIoReq[pThis->iEntry]);
272 pThis->papIoReq[pThis->iEntry] = NULL;
273 }
274 }
275 else
276 RTMemFree(pIoReq);
277}
278
279static void drvdiskintIoLogEntryRelease(PIOLOGENT pIoLogEnt)
280{
281 pIoLogEnt->cRefs--;
282 if (!pIoLogEnt->cRefs)
283 RTMemFree(pIoLogEnt);
284}
285
286/**
287 * Record a successful write to the virtual disk.
288 *
289 * @returns VBox status code.
290 * @param pThis Disk integrity driver instance data.
291 * @param paSeg Segment array of the write to record.
292 * @param cSeg Number of segments.
293 * @param off Start offset.
294 * @param cbWrite Number of bytes to record.
295 */
296static int drvdiskintWriteRecord(PDRVDISKINTEGRITY pThis, PCRTSGSEG paSeg, unsigned cSeg,
297 uint64_t off, size_t cbWrite)
298{
299 int rc = VINF_SUCCESS;
300
301 LogFlowFunc(("pThis=%#p paSeg=%#p cSeg=%u off=%llx cbWrite=%u\n",
302 pThis, paSeg, cSeg, off, cbWrite));
303
304 /* Update the segments */
305 size_t cbLeft = cbWrite;
306 RTFOFF offCurr = (RTFOFF)off;
307 RTSGBUF SgBuf;
308 PIOLOGENT pIoLogEnt = (PIOLOGENT)RTMemAllocZ(sizeof(IOLOGENT));
309 if (!pIoLogEnt)
310 return VERR_NO_MEMORY;
311
312 pIoLogEnt->off = off;
313 pIoLogEnt->cbWrite = cbWrite;
314 pIoLogEnt->cRefs = 0;
315
316 RTSgBufInit(&SgBuf, paSeg, cSeg);
317
318 while (cbLeft)
319 {
320 PDRVDISKSEGMENT pSeg = (PDRVDISKSEGMENT)RTAvlrFileOffsetRangeGet(pThis->pTreeSegments, offCurr);
321 size_t cbRange = 0;
322 bool fSet = false;
323 unsigned offSeg = 0;
324
325 if (!pSeg)
326 {
327 /* Get next segment */
328 pSeg = (PDRVDISKSEGMENT)RTAvlrFileOffsetGetBestFit(pThis->pTreeSegments, offCurr, true);
329 if ( !pSeg
330 || offCurr + (RTFOFF)cbLeft <= pSeg->Core.Key)
331 cbRange = cbLeft;
332 else
333 cbRange = pSeg->Core.Key - offCurr;
334
335 Assert(cbRange % 512 == 0);
336
337 /* Create new segment */
338 pSeg = (PDRVDISKSEGMENT)RTMemAllocZ(RT_OFFSETOF(DRVDISKSEGMENT, apIoLog[cbRange / 512]));
339 if (pSeg)
340 {
341 pSeg->Core.Key = offCurr;
342 pSeg->Core.KeyLast = offCurr + (RTFOFF)cbRange - 1;
343 pSeg->cbSeg = cbRange;
344 pSeg->pbSeg = (uint8_t *)RTMemAllocZ(cbRange);
345 pSeg->cIoLogEntries = cbRange / 512;
346 if (!pSeg->pbSeg)
347 RTMemFree(pSeg);
348 else
349 {
350 bool fInserted = RTAvlrFileOffsetInsert(pThis->pTreeSegments, &pSeg->Core);
351 AssertMsg(fInserted, ("Bug!\n"));
352 fSet = true;
353 }
354 }
355 }
356 else
357 {
358 fSet = true;
359 offSeg = offCurr - pSeg->Core.Key;
360 cbRange = RT_MIN(cbLeft, (size_t)(pSeg->Core.KeyLast + 1 - offCurr));
361 }
362
363 if (fSet)
364 {
365 AssertPtr(pSeg);
366 size_t cbCopied = RTSgBufCopyToBuf(&SgBuf, pSeg->pbSeg + offSeg, cbRange);
367 Assert(cbCopied == cbRange);
368
369 /* Update the I/O log pointers */
370 Assert(offSeg % 512 == 0);
371 Assert(cbRange % 512 == 0);
372 while (offSeg < cbRange)
373 {
374 uint32_t uSector = offSeg / 512;
375 PIOLOGENT pIoLogOld = NULL;
376
377 AssertMsg(uSector < pSeg->cIoLogEntries, ("Internal bug!\n"));
378
379 pIoLogOld = pSeg->apIoLog[uSector];
380 if (pIoLogOld)
381 {
382 pIoLogOld->cRefs--;
383 if (!pIoLogOld->cRefs)
384 RTMemFree(pIoLogOld);
385 }
386
387 pSeg->apIoLog[uSector] = pIoLogEnt;
388 pIoLogEnt->cRefs++;
389
390 offSeg += 512;
391 }
392 }
393 else
394 RTSgBufAdvance(&SgBuf, cbRange);
395
396 offCurr += cbRange;
397 cbLeft -= cbRange;
398 }
399
400 return rc;
401}
402
403/**
404 * Verifies a read request.
405 *
406 * @returns VBox status code.
407 * @param pThis Disk integrity driver instance data.
408 * @param paSeg Segment array of the containing the data buffers to verify.
409 * @param cSeg Number of segments.
410 * @param off Start offset.
411 * @param cbWrite Number of bytes to verify.
412 */
413static int drvdiskintReadVerify(PDRVDISKINTEGRITY pThis, PCRTSGSEG paSeg, unsigned cSeg,
414 uint64_t off, size_t cbRead)
415{
416 int rc = VINF_SUCCESS;
417
418 LogFlowFunc(("pThis=%#p paSeg=%#p cSeg=%u off=%llx cbRead=%u\n",
419 pThis, paSeg, cSeg, off, cbRead));
420
421 Assert(off % 512 == 0);
422 Assert(cbRead % 512 == 0);
423
424 /* Compare read data */
425 size_t cbLeft = cbRead;
426 RTFOFF offCurr = (RTFOFF)off;
427 RTSGBUF SgBuf;
428
429 RTSgBufInit(&SgBuf, paSeg, cSeg);
430
431 while (cbLeft)
432 {
433 PDRVDISKSEGMENT pSeg = (PDRVDISKSEGMENT)RTAvlrFileOffsetRangeGet(pThis->pTreeSegments, offCurr);
434 size_t cbRange = 0;
435 bool fCmp = false;
436 unsigned offSeg = 0;
437
438 if (!pSeg)
439 {
440 /* Get next segment */
441 pSeg = (PDRVDISKSEGMENT)RTAvlrFileOffsetGetBestFit(pThis->pTreeSegments, offCurr, true);
442 if (!pSeg)
443 {
444 /* No data in the tree for this read. Assume everything is ok. */
445 cbRange = cbLeft;
446 }
447 else if (offCurr + (RTFOFF)cbLeft <= pSeg->Core.Key)
448 cbRange = cbLeft;
449 else
450 cbRange = pSeg->Core.Key - offCurr;
451
452 if (pThis->fPrepopulateRamDisk)
453 {
454 /* No segment means everything should be 0 for this part. */
455 if (!RTSgBufIsZero(&SgBuf, cbRange))
456 {
457 RTMsgError("Corrupted disk at offset %llu (expected everything to be 0)!\n",
458 offCurr);
459 RTAssertDebugBreak();
460 }
461 }
462 }
463 else
464 {
465 fCmp = true;
466 offSeg = offCurr - pSeg->Core.Key;
467 cbRange = RT_MIN(cbLeft, (size_t)(pSeg->Core.KeyLast + 1 - offCurr));
468 }
469
470 if (fCmp)
471 {
472 RTSGSEG Seg;
473 RTSGBUF SgBufCmp;
474 size_t cbOff = 0;
475
476 Seg.cbSeg = cbRange;
477 Seg.pvSeg = pSeg->pbSeg + offSeg;
478
479 RTSgBufInit(&SgBufCmp, &Seg, 1);
480 if (RTSgBufCmpEx(&SgBuf, &SgBufCmp, cbRange, &cbOff, true))
481 {
482 /* Corrupted disk, print I/O log entry of the last write which accessed this range. */
483 uint32_t cSector = (offSeg + cbOff) / 512;
484 AssertMsg(cSector < pSeg->cIoLogEntries, ("Internal bug!\n"));
485
486 RTMsgError("Corrupted disk at offset %llu (%u bytes in the current read buffer)!\n",
487 offCurr + cbOff, cbOff);
488 RTMsgError("Last write to this sector started at offset %llu with %u bytes (%u references to this log entry)\n",
489 pSeg->apIoLog[cSector]->off,
490 pSeg->apIoLog[cSector]->cbWrite,
491 pSeg->apIoLog[cSector]->cRefs);
492 RTAssertDebugBreak();
493 }
494 }
495 else
496 RTSgBufAdvance(&SgBuf, cbRange);
497
498 offCurr += cbRange;
499 cbLeft -= cbRange;
500 }
501
502 return rc;
503}
504
505/**
506 * Discards the given ranges from the disk.
507 *
508 * @returns VBox status code.
509 * @param pThis Disk integrity driver instance data.
510 * @param paRanges Array of ranges to discard.
511 * @param cRanges Number of ranges in the array.
512 */
513static int drvdiskintDiscardRecords(PDRVDISKINTEGRITY pThis, PCRTRANGE paRanges, unsigned cRanges)
514{
515 int rc = VINF_SUCCESS;
516
517 LogFlowFunc(("pThis=%#p paRanges=%#p cRanges=%u\n", pThis, paRanges, cRanges));
518
519 for (unsigned i = 0; i < cRanges; i++)
520 {
521 uint64_t offStart = paRanges[i].offStart;
522 size_t cbLeft = paRanges[i].cbRange;
523
524 LogFlowFunc(("Discarding off=%llu cbRange=%zu\n", offStart, cbLeft));
525
526 while (cbLeft)
527 {
528 size_t cbRange;
529 PDRVDISKSEGMENT pSeg = (PDRVDISKSEGMENT)RTAvlrFileOffsetRangeGet(pThis->pTreeSegments, offStart);
530
531 if (!pSeg)
532 {
533 /* Get next segment */
534 pSeg = (PDRVDISKSEGMENT)RTAvlrFileOffsetGetBestFit(pThis->pTreeSegments, offStart, true);
535 if ( !pSeg
536 || (RTFOFF)offStart + (RTFOFF)cbLeft <= pSeg->Core.Key)
537 cbRange = cbLeft;
538 else
539 cbRange = pSeg->Core.Key - offStart;
540
541 Assert(!(cbRange % 512));
542 }
543 else
544 {
545 size_t cbPreLeft, cbPostLeft;
546
547 cbRange = RT_MIN(cbLeft, pSeg->Core.KeyLast - offStart + 1);
548 cbPreLeft = offStart - pSeg->Core.Key;
549 cbPostLeft = pSeg->cbSeg - cbRange - cbPreLeft;
550
551 Assert(!(cbRange % 512));
552 Assert(!(cbPreLeft % 512));
553 Assert(!(cbPostLeft % 512));
554
555 LogFlowFunc(("cbRange=%zu cbPreLeft=%zu cbPostLeft=%zu\n",
556 cbRange, cbPreLeft, cbPostLeft));
557
558 RTAvlrFileOffsetRemove(pThis->pTreeSegments, pSeg->Core.Key);
559
560 if (!cbPreLeft && !cbPostLeft)
561 {
562 /* Just free the whole segment. */
563 LogFlowFunc(("Freeing whole segment pSeg=%#p\n", pSeg));
564 RTMemFree(pSeg->pbSeg);
565 for (unsigned idx = 0; idx < pSeg->cIoLogEntries; idx++)
566 drvdiskintIoLogEntryRelease(pSeg->apIoLog[idx]);
567 RTMemFree(pSeg);
568 }
569 else if (cbPreLeft && !cbPostLeft)
570 {
571 /* Realloc to new size and insert. */
572 LogFlowFunc(("Realloc segment pSeg=%#p\n", pSeg));
573 pSeg->pbSeg = (uint8_t *)RTMemRealloc(pSeg->pbSeg, cbPreLeft);
574 for (unsigned idx = cbPreLeft / 512; idx < pSeg->cIoLogEntries; idx++)
575 drvdiskintIoLogEntryRelease(pSeg->apIoLog[idx]);
576 pSeg = (PDRVDISKSEGMENT)RTMemRealloc(pSeg, RT_OFFSETOF(DRVDISKSEGMENT, apIoLog[cbPreLeft / 512]));
577 pSeg->Core.KeyLast = pSeg->Core.Key + cbPreLeft - 1;
578 pSeg->cbSeg = cbPreLeft;
579 pSeg->cIoLogEntries = cbPreLeft / 512;
580 bool fInserted = RTAvlrFileOffsetInsert(pThis->pTreeSegments, &pSeg->Core);
581 Assert(fInserted);
582 }
583 else if (!cbPreLeft && cbPostLeft)
584 {
585 /* Move data to the front and realloc. */
586 LogFlowFunc(("Move data and realloc segment pSeg=%#p\n", pSeg));
587 memmove(pSeg->pbSeg, pSeg->pbSeg + cbRange, cbPostLeft);
588 for (unsigned idx = 0; idx < cbRange / 512; idx++)
589 drvdiskintIoLogEntryRelease(pSeg->apIoLog[idx]);
590 for (unsigned idx = 0; idx < cbPostLeft /512; idx++)
591 pSeg->apIoLog[idx] = pSeg->apIoLog[(cbRange / 512) + idx];
592 pSeg = (PDRVDISKSEGMENT)RTMemRealloc(pSeg, RT_OFFSETOF(DRVDISKSEGMENT, apIoLog[cbPostLeft / 512]));
593 pSeg->pbSeg = (uint8_t *)RTMemRealloc(pSeg->pbSeg, cbPostLeft);
594 pSeg->Core.Key += cbRange;
595 pSeg->cbSeg = cbPostLeft;
596 pSeg->cIoLogEntries = cbPostLeft / 512;
597 bool fInserted = RTAvlrFileOffsetInsert(pThis->pTreeSegments, &pSeg->Core);
598 Assert(fInserted);
599 }
600 else
601 {
602 /* Split the segment into 2 new segments. */
603 LogFlowFunc(("Split segment pSeg=%#p\n", pSeg));
604 PDRVDISKSEGMENT pSegPost = (PDRVDISKSEGMENT)RTMemAllocZ(RT_OFFSETOF(DRVDISKSEGMENT, apIoLog[cbPostLeft / 512]));
605 if (pSegPost)
606 {
607 pSegPost->Core.Key = pSeg->Core.Key + cbPreLeft + cbRange;
608 pSegPost->Core.KeyLast = pSeg->Core.KeyLast;
609 pSegPost->cbSeg = cbPostLeft;
610 pSegPost->pbSeg = (uint8_t *)RTMemAllocZ(cbPostLeft);
611 pSegPost->cIoLogEntries = cbPostLeft / 512;
612 if (!pSegPost->pbSeg)
613 RTMemFree(pSegPost);
614 else
615 {
616 memcpy(pSegPost->pbSeg, pSeg->pbSeg + cbPreLeft + cbRange, cbPostLeft);
617 for (unsigned idx = 0; idx < cbPostLeft / 512; idx++)
618 pSegPost->apIoLog[idx] = pSeg->apIoLog[((cbPreLeft + cbRange) / 512) + idx];
619
620 bool fInserted = RTAvlrFileOffsetInsert(pThis->pTreeSegments, &pSegPost->Core);
621 Assert(fInserted);
622 }
623 }
624
625 /* Shrink the current segment. */
626 pSeg->pbSeg = (uint8_t *)RTMemRealloc(pSeg->pbSeg, cbPreLeft);
627 for (unsigned idx = cbPreLeft / 512; idx < (cbPreLeft + cbRange) / 512; idx++)
628 drvdiskintIoLogEntryRelease(pSeg->apIoLog[idx]);
629 pSeg = (PDRVDISKSEGMENT)RTMemRealloc(pSeg, RT_OFFSETOF(DRVDISKSEGMENT, apIoLog[cbPreLeft / 512]));
630 pSeg->Core.KeyLast = pSeg->Core.Key + cbPreLeft - 1;
631 pSeg->cbSeg = cbPreLeft;
632 pSeg->cIoLogEntries = cbPreLeft / 512;
633 bool fInserted = RTAvlrFileOffsetInsert(pThis->pTreeSegments, &pSeg->Core);
634 Assert(fInserted);
635 } /* if (cbPreLeft && cbPostLeft) */
636 }
637
638 offStart += cbRange;
639 cbLeft -= cbRange;
640 }
641 }
642
643 LogFlowFunc(("returns rc=%Rrc\n", rc));
644 return rc;
645}
646
647/**
648 * Adds a request to the active list.
649 *
650 * @returns nothing.
651 * @param pThis The driver instance data.
652 * @param pIoReq The request to add.
653 */
654static void drvdiskintIoReqAdd(PDRVDISKINTEGRITY pThis, PDRVDISKAIOREQ pIoReq)
655{
656 PDRVDISKAIOREQACTIVE pReqActive = &pThis->apReqActive[pThis->iNextFreeSlot];
657
658 Assert(!pReqActive->pIoReq);
659 pReqActive->tsStart = pIoReq->tsStart;
660 pReqActive->pIoReq = pIoReq;
661 pIoReq->iSlot = pThis->iNextFreeSlot;
662
663 /* Search for the next one. */
664 while (pThis->apReqActive[pThis->iNextFreeSlot].pIoReq)
665 pThis->iNextFreeSlot = (pThis->iNextFreeSlot+1) % RT_ELEMENTS(pThis->apReqActive);
666}
667
668/**
669 * Removes a request from the active list.
670 *
671 * @returns nothing.
672 * @param pThis The driver instance data.
673 * @param pIoReq The request to remove.
674 */
675static void drvdiskintIoReqRemove(PDRVDISKINTEGRITY pThis, PDRVDISKAIOREQ pIoReq)
676{
677 PDRVDISKAIOREQACTIVE pReqActive = &pThis->apReqActive[pIoReq->iSlot];
678
679 Assert(pReqActive->pIoReq == pIoReq);
680
681 ASMAtomicWriteNullPtr(&pReqActive->pIoReq);
682}
683
684/**
685 * Thread checking for expired requests.
686 *
687 * @returns IPRT status code.
688 * @param pThread Thread handle.
689 * @param pvUser Opaque user data.
690 */
691static int drvdiskIntIoReqExpiredCheck(RTTHREAD pThread, void *pvUser)
692{
693 PDRVDISKINTEGRITY pThis = (PDRVDISKINTEGRITY)pvUser;
694
695 while (pThis->fRunning)
696 {
697 int rc = RTSemEventWait(pThis->SemEvent, pThis->uCheckIntervalMs);
698
699 if (!pThis->fRunning)
700 break;
701
702 Assert(rc == VERR_TIMEOUT);
703
704 /* Get current timestamp for comparison. */
705 uint64_t tsCurr = RTTimeSystemMilliTS();
706
707 /* Go through the array and check for expired requests. */
708 for (unsigned i = 0; i < RT_ELEMENTS(pThis->apReqActive); i++)
709 {
710 PDRVDISKAIOREQACTIVE pReqActive = &pThis->apReqActive[i];
711 PDRVDISKAIOREQ pIoReq = ASMAtomicReadPtrT(&pReqActive->pIoReq, PDRVDISKAIOREQ);
712
713 if ( pIoReq
714 && (tsCurr > pReqActive->tsStart)
715 && (tsCurr - pReqActive->tsStart) >= pThis->uExpireIntervalMs)
716 {
717 RTMsgError("Request %#p expired (active for %llu ms already)\n",
718 pIoReq, tsCurr - pReqActive->tsStart);
719 RTAssertDebugBreak();
720 }
721 }
722 }
723
724 return VINF_SUCCESS;
725}
726
727/**
728 * Verify a completed read after write request.
729 *
730 * @returns VBox status code.
731 * @param pThis The driver instance data.
732 * @param pIoReq The request to be verified.
733 */
734static int drvdiskintReadAfterWriteVerify(PDRVDISKINTEGRITY pThis, PDRVDISKAIOREQ pIoReq)
735{
736 int rc = VINF_SUCCESS;
737
738 if (pThis->fCheckConsistency)
739 rc = drvdiskintReadVerify(pThis, pIoReq->paSeg, pIoReq->cSeg, pIoReq->off, pIoReq->cbTransfer);
740 else /** @todo: Implement read after write verification without a memory based image of the disk. */
741 AssertMsgFailed(("TODO\n"));
742
743 return rc;
744}
745
746/* -=-=-=-=- IMedia -=-=-=-=- */
747
748/** Makes a PDRVDISKINTEGRITY out of a PPDMIMEDIA. */
749#define PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface) ( (PDRVDISKINTEGRITY)((uintptr_t)pInterface - RT_OFFSETOF(DRVDISKINTEGRITY, IMedia)) )
750/** Makes a PDRVDISKINTEGRITY out of a PPDMIMEDIAASYNC. */
751#define PDMIMEDIAASYNC_2_DRVDISKINTEGRITY(pInterface) ( (PDRVDISKINTEGRITY)((uintptr_t)pInterface - RT_OFFSETOF(DRVDISKINTEGRITY, IMediaAsync)) )
752
753
754/*********************************************************************************************************************************
755* Media interface methods *
756*********************************************************************************************************************************/
757
758/** @copydoc PDMIMEDIA::pfnRead */
759static DECLCALLBACK(int) drvdiskintRead(PPDMIMEDIA pInterface,
760 uint64_t off, void *pvBuf, size_t cbRead)
761{
762 int rc = VINF_SUCCESS;
763 VDIOLOGENT hIoLogEntry;
764 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
765
766 if (pThis->hIoLogger)
767 {
768 rc = VDDbgIoLogStart(pThis->hIoLogger, false, VDDBGIOLOGREQ_READ, off,
769 cbRead, NULL, &hIoLogEntry);
770 AssertRC(rc);
771 }
772
773 rc = pThis->pDrvMedia->pfnRead(pThis->pDrvMedia, off, pvBuf, cbRead);
774
775 if (pThis->hIoLogger)
776 {
777 RTSGSEG Seg;
778 RTSGBUF SgBuf;
779
780 Seg.pvSeg = pvBuf;
781 Seg.cbSeg = cbRead;
782 RTSgBufInit(&SgBuf, &Seg, 1);
783
784 int rc2 = VDDbgIoLogComplete(pThis->hIoLogger, hIoLogEntry, rc, &SgBuf);
785 AssertRC(rc2);
786 }
787
788 if (RT_FAILURE(rc))
789 return rc;
790
791 if (pThis->fCheckConsistency)
792 {
793 /* Verify the read. */
794 RTSGSEG Seg;
795 Seg.cbSeg = cbRead;
796 Seg.pvSeg = pvBuf;
797 rc = drvdiskintReadVerify(pThis, &Seg, 1, off, cbRead);
798 }
799
800 return rc;
801}
802
803/** @copydoc PDMIMEDIA::pfnWrite */
804static DECLCALLBACK(int) drvdiskintWrite(PPDMIMEDIA pInterface,
805 uint64_t off, const void *pvBuf,
806 size_t cbWrite)
807{
808 int rc = VINF_SUCCESS;
809 VDIOLOGENT hIoLogEntry;
810 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
811
812 if (pThis->hIoLogger)
813 {
814 RTSGSEG Seg;
815 RTSGBUF SgBuf;
816
817 Seg.pvSeg = (void *)pvBuf;
818 Seg.cbSeg = cbWrite;
819 RTSgBufInit(&SgBuf, &Seg, 1);
820
821 rc = VDDbgIoLogStart(pThis->hIoLogger, false, VDDBGIOLOGREQ_WRITE, off,
822 cbWrite, &SgBuf, &hIoLogEntry);
823 AssertRC(rc);
824 }
825
826 if (pThis->fRecordWriteBeforeCompletion)
827 {
828 RTSGSEG Seg;
829 Seg.cbSeg = cbWrite;
830 Seg.pvSeg = (void *)pvBuf;
831
832 rc = drvdiskintWriteRecord(pThis, &Seg, 1, off, cbWrite);
833 if (RT_FAILURE(rc))
834 return rc;
835 }
836
837 rc = pThis->pDrvMedia->pfnWrite(pThis->pDrvMedia, off, pvBuf, cbWrite);
838
839 if (pThis->hIoLogger)
840 {
841 int rc2 = VDDbgIoLogComplete(pThis->hIoLogger, hIoLogEntry, rc, NULL);
842 AssertRC(rc2);
843 }
844
845 if (RT_FAILURE(rc))
846 return rc;
847
848 if ( pThis->fCheckConsistency
849 && !pThis->fRecordWriteBeforeCompletion)
850 {
851 /* Record the write. */
852 RTSGSEG Seg;
853 Seg.cbSeg = cbWrite;
854 Seg.pvSeg = (void *)pvBuf;
855 rc = drvdiskintWriteRecord(pThis, &Seg, 1, off, cbWrite);
856 }
857
858 return rc;
859}
860
861static DECLCALLBACK(int) drvdiskintStartRead(PPDMIMEDIAASYNC pInterface, uint64_t uOffset,
862 PCRTSGSEG paSeg, unsigned cSeg,
863 size_t cbRead, void *pvUser)
864{
865 LogFlow(("%s: uOffset=%llu paSeg=%#p cSeg=%u cbRead=%d pvUser=%#p\n", __FUNCTION__,
866 uOffset, paSeg, cSeg, cbRead, pvUser));
867 PDRVDISKINTEGRITY pThis = PDMIMEDIAASYNC_2_DRVDISKINTEGRITY(pInterface);
868 PDRVDISKAIOREQ pIoReq = drvdiskintIoReqAlloc(DRVDISKAIOTXDIR_READ, uOffset, paSeg, cSeg, cbRead, pvUser);
869 AssertPtr(pIoReq);
870
871 if (pThis->fTraceRequests)
872 drvdiskintIoReqAdd(pThis, pIoReq);
873
874 if (pThis->hIoLogger)
875 {
876 int rc2 = VDDbgIoLogStart(pThis->hIoLogger, true, VDDBGIOLOGREQ_READ, uOffset,
877 cbRead, NULL, &pIoReq->hIoLogEntry);
878 AssertRC(rc2);
879 }
880
881 int rc = pThis->pDrvMediaAsync->pfnStartRead(pThis->pDrvMediaAsync, uOffset, paSeg, cSeg,
882 cbRead, pIoReq);
883 if (rc == VINF_VD_ASYNC_IO_FINISHED)
884 {
885 /* Verify the read now. */
886 if (pThis->fCheckConsistency)
887 {
888 int rc2 = drvdiskintReadVerify(pThis, paSeg, cSeg, uOffset, cbRead);
889 AssertRC(rc2);
890 }
891
892 if (pThis->hIoLogger)
893 {
894 RTSGBUF SgBuf;
895
896 RTSgBufInit(&SgBuf, paSeg, cSeg);
897
898 int rc2 = VDDbgIoLogComplete(pThis->hIoLogger, pIoReq->hIoLogEntry, VINF_SUCCESS, &SgBuf);
899 AssertRC(rc2);
900 }
901
902 if (pThis->fTraceRequests)
903 drvdiskintIoReqRemove(pThis, pIoReq);
904 RTMemFree(pIoReq);
905 }
906 else if (RT_FAILURE(rc) && rc != VERR_VD_ASYNC_IO_IN_PROGRESS)
907 RTMemFree(pIoReq);
908
909 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
910 return rc;
911}
912
913static DECLCALLBACK(int) drvdiskintStartWrite(PPDMIMEDIAASYNC pInterface, uint64_t uOffset,
914 PCRTSGSEG paSeg, unsigned cSeg,
915 size_t cbWrite, void *pvUser)
916{
917 LogFlow(("%s: uOffset=%#llx paSeg=%#p cSeg=%u cbWrite=%d pvUser=%#p\n", __FUNCTION__,
918 uOffset, paSeg, cSeg, cbWrite, pvUser));
919 PDRVDISKINTEGRITY pThis = PDMIMEDIAASYNC_2_DRVDISKINTEGRITY(pInterface);
920 PDRVDISKAIOREQ pIoReq = drvdiskintIoReqAlloc(DRVDISKAIOTXDIR_WRITE, uOffset, paSeg, cSeg, cbWrite, pvUser);
921 AssertPtr(pIoReq);
922
923 if (pThis->fTraceRequests)
924 drvdiskintIoReqAdd(pThis, pIoReq);
925
926 if (pThis->hIoLogger)
927 {
928 RTSGBUF SgBuf;
929
930 RTSgBufInit(&SgBuf, paSeg, cSeg);
931
932 int rc2 = VDDbgIoLogStart(pThis->hIoLogger, true, VDDBGIOLOGREQ_WRITE, uOffset,
933 cbWrite, &SgBuf, &pIoReq->hIoLogEntry);
934 AssertRC(rc2);
935 }
936
937 if (pThis->fRecordWriteBeforeCompletion)
938 {
939 int rc2 = drvdiskintWriteRecord(pThis, paSeg, cSeg, uOffset, cbWrite);
940 AssertRC(rc2);
941 }
942
943 int rc = pThis->pDrvMediaAsync->pfnStartWrite(pThis->pDrvMediaAsync, uOffset, paSeg, cSeg,
944 cbWrite, pIoReq);
945 if (rc == VINF_VD_ASYNC_IO_FINISHED)
946 {
947 /* Record the write. */
948 if ( pThis->fCheckConsistency
949 && !pThis->fRecordWriteBeforeCompletion)
950 {
951 int rc2 = drvdiskintWriteRecord(pThis, paSeg, cSeg, uOffset, cbWrite);
952 AssertRC(rc2);
953 }
954
955 if (pThis->hIoLogger)
956 {
957 int rc2 = VDDbgIoLogComplete(pThis->hIoLogger, pIoReq->hIoLogEntry, VINF_SUCCESS, NULL);
958 AssertRC(rc2);
959 }
960
961 if (pThis->fTraceRequests)
962 drvdiskintIoReqRemove(pThis, pIoReq);
963
964 RTMemFree(pIoReq);
965 }
966 else if (RT_FAILURE(rc) && rc != VERR_VD_ASYNC_IO_IN_PROGRESS)
967 RTMemFree(pIoReq);
968
969 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
970 return rc;
971}
972
973/** @copydoc PDMIMEDIAASYNC::pfnStartFlush */
974static DECLCALLBACK(int) drvdiskintStartFlush(PPDMIMEDIAASYNC pInterface, void *pvUser)
975{
976 int rc = VINF_SUCCESS;
977 PDRVDISKINTEGRITY pThis = PDMIMEDIAASYNC_2_DRVDISKINTEGRITY(pInterface);
978 PDRVDISKAIOREQ pIoReq = drvdiskintIoReqAlloc(DRVDISKAIOTXDIR_FLUSH, 0, NULL, 0, 0, pvUser);
979 AssertPtr(pIoReq);
980
981 if (pThis->fTraceRequests)
982 drvdiskintIoReqAdd(pThis, pIoReq);
983
984 if (pThis->hIoLogger)
985 {
986 rc = VDDbgIoLogStart(pThis->hIoLogger, true, VDDBGIOLOGREQ_FLUSH, 0,
987 0, NULL, &pIoReq->hIoLogEntry);
988 AssertRC(rc);
989 }
990
991 rc = pThis->pDrvMediaAsync->pfnStartFlush(pThis->pDrvMediaAsync, pIoReq);
992
993 if (rc == VINF_VD_ASYNC_IO_FINISHED)
994 {
995 if (pThis->hIoLogger)
996 {
997 int rc2 = VDDbgIoLogComplete(pThis->hIoLogger, pIoReq->hIoLogEntry, VINF_SUCCESS, NULL);
998 AssertRC(rc2);
999 }
1000
1001 RTMemFree(pIoReq);
1002 }
1003 else if (RT_FAILURE(rc) && rc != VERR_VD_ASYNC_IO_IN_PROGRESS)
1004 RTMemFree(pIoReq);
1005
1006 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
1007 return rc;
1008}
1009
1010/** @copydoc PDMIMEDIAASYNC::pfnStartDiscard */
1011static DECLCALLBACK(int) drvdiskintStartDiscard(PPDMIMEDIAASYNC pInterface, PCRTRANGE paRanges, unsigned cRanges, void *pvUser)
1012{
1013 int rc = VINF_SUCCESS;
1014 VDIOLOGENT hIoLogEntry;
1015 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1016 PDRVDISKAIOREQ pIoReq = drvdiskintIoReqAlloc(DRVDISKAIOTXDIR_DISCARD, 0, NULL, 0, 0, pvUser);
1017 AssertPtr(pIoReq);
1018
1019 pIoReq->paRanges = paRanges;
1020 pIoReq->cRanges = cRanges;
1021
1022 if (pThis->hIoLogger)
1023 {
1024 rc = VDDbgIoLogStartDiscard(pThis->hIoLogger, true, paRanges, cRanges, &hIoLogEntry);
1025 AssertRC(rc);
1026 }
1027
1028 rc = pThis->pDrvMediaAsync->pfnStartDiscard(pThis->pDrvMediaAsync, paRanges, cRanges, pIoReq);
1029
1030 if (rc == VINF_VD_ASYNC_IO_FINISHED)
1031 {
1032 if (pThis->hIoLogger)
1033 {
1034 int rc2 = VDDbgIoLogComplete(pThis->hIoLogger, pIoReq->hIoLogEntry, VINF_SUCCESS, NULL);
1035 AssertRC(rc2);
1036 }
1037
1038 RTMemFree(pIoReq);
1039 }
1040 else if (RT_FAILURE(rc) && rc != VERR_VD_ASYNC_IO_IN_PROGRESS)
1041 RTMemFree(pIoReq);
1042
1043 return rc;
1044}
1045
1046/** @copydoc PDMIMEDIA::pfnFlush */
1047static DECLCALLBACK(int) drvdiskintFlush(PPDMIMEDIA pInterface)
1048{
1049 int rc = VINF_SUCCESS;
1050 VDIOLOGENT hIoLogEntry;
1051 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1052
1053 if (pThis->hIoLogger)
1054 {
1055 rc = VDDbgIoLogStart(pThis->hIoLogger, false, VDDBGIOLOGREQ_FLUSH, 0,
1056 0, NULL, &hIoLogEntry);
1057 AssertRC(rc);
1058 }
1059
1060 rc = pThis->pDrvMedia->pfnFlush(pThis->pDrvMedia);
1061
1062 if (pThis->hIoLogger)
1063 {
1064 int rc2 = VDDbgIoLogComplete(pThis->hIoLogger, hIoLogEntry, rc, NULL);
1065 AssertRC(rc2);
1066 }
1067
1068 return rc;
1069}
1070
1071/** @copydoc PDMIMEDIA::pfnGetSize */
1072static DECLCALLBACK(uint64_t) drvdiskintGetSize(PPDMIMEDIA pInterface)
1073{
1074 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1075 return pThis->pDrvMedia->pfnGetSize(pThis->pDrvMedia);
1076}
1077
1078/** @copydoc PDMIMEDIA::pfnIsReadOnly */
1079static DECLCALLBACK(bool) drvdiskintIsReadOnly(PPDMIMEDIA pInterface)
1080{
1081 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1082 return pThis->pDrvMedia->pfnIsReadOnly(pThis->pDrvMedia);
1083}
1084
1085/** @copydoc PDMIMEDIA::pfnBiosGetPCHSGeometry */
1086static DECLCALLBACK(int) drvdiskintBiosGetPCHSGeometry(PPDMIMEDIA pInterface,
1087 PPDMMEDIAGEOMETRY pPCHSGeometry)
1088{
1089 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1090 return pThis->pDrvMedia->pfnBiosGetPCHSGeometry(pThis->pDrvMedia, pPCHSGeometry);
1091}
1092
1093/** @copydoc PDMIMEDIA::pfnBiosSetPCHSGeometry */
1094static DECLCALLBACK(int) drvdiskintBiosSetPCHSGeometry(PPDMIMEDIA pInterface,
1095 PCPDMMEDIAGEOMETRY pPCHSGeometry)
1096{
1097 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1098 return pThis->pDrvMedia->pfnBiosSetPCHSGeometry(pThis->pDrvMedia, pPCHSGeometry);
1099}
1100
1101/** @copydoc PDMIMEDIA::pfnBiosGetLCHSGeometry */
1102static DECLCALLBACK(int) drvdiskintBiosGetLCHSGeometry(PPDMIMEDIA pInterface,
1103 PPDMMEDIAGEOMETRY pLCHSGeometry)
1104{
1105 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1106 return pThis->pDrvMedia->pfnBiosGetLCHSGeometry(pThis->pDrvMedia, pLCHSGeometry);
1107}
1108
1109/** @copydoc PDMIMEDIA::pfnBiosSetLCHSGeometry */
1110static DECLCALLBACK(int) drvdiskintBiosSetLCHSGeometry(PPDMIMEDIA pInterface,
1111 PCPDMMEDIAGEOMETRY pLCHSGeometry)
1112{
1113 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1114 return pThis->pDrvMedia->pfnBiosSetLCHSGeometry(pThis->pDrvMedia, pLCHSGeometry);
1115}
1116
1117/** @copydoc PDMIMEDIA::pfnGetUuid */
1118static DECLCALLBACK(int) drvdiskintGetUuid(PPDMIMEDIA pInterface, PRTUUID pUuid)
1119{
1120 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1121 return pThis->pDrvMedia->pfnGetUuid(pThis->pDrvMedia, pUuid);
1122}
1123
1124/** @copydoc PDMIMEDIA::pfnGetSectorSize */
1125static DECLCALLBACK(uint32_t) drvdiskintGetSectorSize(PPDMIMEDIA pInterface)
1126{
1127 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1128 return pThis->pDrvMedia->pfnGetSectorSize(pThis->pDrvMedia);
1129}
1130
1131/** @copydoc PDMIMEDIA::pfnDiscard */
1132static DECLCALLBACK(int) drvdiskintDiscard(PPDMIMEDIA pInterface, PCRTRANGE paRanges, unsigned cRanges)
1133{
1134 int rc = VINF_SUCCESS;
1135 VDIOLOGENT hIoLogEntry;
1136 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1137
1138 if (pThis->hIoLogger)
1139 {
1140 rc = VDDbgIoLogStartDiscard(pThis->hIoLogger, false, paRanges, cRanges, &hIoLogEntry);
1141 AssertRC(rc);
1142 }
1143
1144 rc = pThis->pDrvMedia->pfnDiscard(pThis->pDrvMedia, paRanges, cRanges);
1145
1146 if (pThis->hIoLogger)
1147 {
1148 int rc2 = VDDbgIoLogComplete(pThis->hIoLogger, hIoLogEntry, rc, NULL);
1149 AssertRC(rc2);
1150 }
1151
1152 if (pThis->fCheckConsistency)
1153 rc = drvdiskintDiscardRecords(pThis, paRanges, cRanges);
1154
1155 return rc;
1156}
1157
1158/** @copydoc PDMIMEDIA::pfnIoBufAlloc */
1159static DECLCALLBACK(int) drvdiskintIoBufAlloc(PPDMIMEDIA pInterface, size_t cb, void **ppvNew)
1160{
1161 LogFlowFunc(("\n"));
1162 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1163
1164 return pThis->pDrvMedia->pfnIoBufAlloc(pThis->pDrvMedia, cb, ppvNew);
1165}
1166
1167/** @copydoc PDMIMEDIA::pfnIoBufFree */
1168static DECLCALLBACK(int) drvdiskintIoBufFree(PPDMIMEDIA pInterface, void *pv, size_t cb)
1169{
1170 LogFlowFunc(("\n"));
1171 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1172
1173 return pThis->pDrvMedia->pfnIoBufFree(pThis->pDrvMedia, pv, cb);
1174}
1175
1176/** @copydoc PDMIMEDIA::pfnReadPcBios */
1177static DECLCALLBACK(int) drvdiskintReadPcBios(PPDMIMEDIA pInterface,
1178 uint64_t off, void *pvBuf, size_t cbRead)
1179{
1180 LogFlowFunc(("\n"));
1181 PDRVDISKINTEGRITY pThis = PDMIMEDIA_2_DRVDISKINTEGRITY(pInterface);
1182
1183 return pThis->pDrvMedia->pfnReadPcBios(pThis->pDrvMedia, off, pvBuf, cbRead);
1184}
1185
1186/* -=-=-=-=- IMediaAsyncPort -=-=-=-=- */
1187
1188/** Makes a PDRVBLOCKASYNC out of a PPDMIMEDIAASYNCPORT. */
1189#define PDMIMEDIAASYNCPORT_2_DRVDISKINTEGRITY(pInterface) ( (PDRVDISKINTEGRITY((uintptr_t)pInterface - RT_OFFSETOF(DRVDISKINTEGRITY, IMediaAsyncPort))) )
1190
1191static DECLCALLBACK(int) drvdiskintAsyncTransferCompleteNotify(PPDMIMEDIAASYNCPORT pInterface, void *pvUser, int rcReq)
1192{
1193 PDRVDISKINTEGRITY pThis = PDMIMEDIAASYNCPORT_2_DRVDISKINTEGRITY(pInterface);
1194 PDRVDISKAIOREQ pIoReq = (PDRVDISKAIOREQ)pvUser;
1195 int rc = VINF_SUCCESS;
1196
1197 LogFlowFunc(("pIoReq=%#p\n", pIoReq));
1198
1199 /* Remove from the active list. */
1200 if (pThis->fTraceRequests)
1201 drvdiskintIoReqRemove(pThis, pIoReq);
1202
1203 if (RT_SUCCESS(rcReq) && pThis->fCheckConsistency)
1204 {
1205 if (pIoReq->enmTxDir == DRVDISKAIOTXDIR_READ)
1206 rc = drvdiskintReadVerify(pThis, pIoReq->paSeg, pIoReq->cSeg, pIoReq->off, pIoReq->cbTransfer);
1207 else if ( pIoReq->enmTxDir == DRVDISKAIOTXDIR_WRITE
1208 && !pThis->fRecordWriteBeforeCompletion)
1209 rc = drvdiskintWriteRecord(pThis, pIoReq->paSeg, pIoReq->cSeg, pIoReq->off, pIoReq->cbTransfer);
1210 else if (pIoReq->enmTxDir == DRVDISKAIOTXDIR_DISCARD)
1211 rc = drvdiskintDiscardRecords(pThis, pIoReq->paRanges, pIoReq->cRanges);
1212 else if (pIoReq->enmTxDir == DRVDISKAIOTXDIR_READ_AFTER_WRITE)
1213 rc = drvdiskintReadAfterWriteVerify(pThis, pIoReq);
1214 else
1215 AssertMsg( pIoReq->enmTxDir == DRVDISKAIOTXDIR_FLUSH
1216 || ( pIoReq->enmTxDir == DRVDISKAIOTXDIR_WRITE
1217 && pThis->fRecordWriteBeforeCompletion), ("Huh?\n"));
1218
1219 AssertRC(rc);
1220 }
1221
1222 if (pThis->hIoLogger)
1223 {
1224 RTSGBUF SgBuf;
1225
1226 if (pIoReq->enmTxDir == DRVDISKAIOTXDIR_READ)
1227 RTSgBufInit(&SgBuf, pIoReq->paSeg, pIoReq->cSeg);
1228
1229 int rc2 = VDDbgIoLogComplete(pThis->hIoLogger, pIoReq->hIoLogEntry, rc, &SgBuf);
1230 AssertRC(rc2);
1231 }
1232
1233 if ( pThis->fReadAfterWrite
1234 && pIoReq->enmTxDir == DRVDISKAIOTXDIR_WRITE)
1235 {
1236 pIoReq->enmTxDir = DRVDISKAIOTXDIR_READ_AFTER_WRITE;
1237
1238 /* Readd because it was rmeoved above. */
1239 if (pThis->fTraceRequests)
1240 drvdiskintIoReqAdd(pThis, pIoReq);
1241
1242 rc = pThis->pDrvMediaAsync->pfnStartRead(pThis->pDrvMediaAsync, pIoReq->off, pIoReq->paSeg, pIoReq->cSeg,
1243 pIoReq->cbTransfer, pIoReq);
1244 if (rc == VINF_VD_ASYNC_IO_FINISHED)
1245 {
1246 rc = drvdiskintReadAfterWriteVerify(pThis, pIoReq);
1247
1248 if (pThis->fTraceRequests)
1249 drvdiskintIoReqRemove(pThis, pIoReq);
1250 RTMemFree(pIoReq);
1251 }
1252 else if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
1253 rc = VINF_SUCCESS;
1254 else if (RT_FAILURE(rc))
1255 RTMemFree(pIoReq);
1256 }
1257 else
1258 {
1259 void *pvUserComplete = pIoReq->pvUser;
1260 drvdiskintIoReqFree(pThis, pIoReq);
1261
1262 rc = pThis->pDrvMediaAsyncPort->pfnTransferCompleteNotify(pThis->pDrvMediaAsyncPort, pvUserComplete, rcReq);
1263 }
1264
1265 return rc;
1266}
1267
1268/* -=-=-=-=- IMediaPort -=-=-=-=- */
1269
1270/** Makes a PDRVBLOCK out of a PPDMIMEDIAPORT. */
1271#define PDMIMEDIAPORT_2_DRVDISKINTEGRITY(pInterface) ( (PDRVDISKINTEGRITY((uintptr_t)pInterface - RT_OFFSETOF(DRVDISKINTEGRITY, IMediaPort))) )
1272
1273/**
1274 * @interface_method_impl{PDMIMEDIAPORT,pfnQueryDeviceLocation}
1275 */
1276static DECLCALLBACK(int) drvdiskintQueryDeviceLocation(PPDMIMEDIAPORT pInterface, const char **ppcszController,
1277 uint32_t *piInstance, uint32_t *piLUN)
1278{
1279 PDRVDISKINTEGRITY pThis = PDMIMEDIAPORT_2_DRVDISKINTEGRITY(pInterface);
1280
1281 return pThis->pDrvMediaPort->pfnQueryDeviceLocation(pThis->pDrvMediaPort, ppcszController,
1282 piInstance, piLUN);
1283}
1284
1285/* -=-=-=-=- IBase -=-=-=-=- */
1286
1287/**
1288 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1289 */
1290static DECLCALLBACK(void *) drvdiskintQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1291{
1292 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
1293 PDRVDISKINTEGRITY pThis = PDMINS_2_DATA(pDrvIns, PDRVDISKINTEGRITY);
1294
1295 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
1296 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIA, &pThis->IMedia);
1297 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAASYNC, pThis->pDrvMediaAsync ? &pThis->IMediaAsync : NULL);
1298 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAASYNCPORT, &pThis->IMediaAsyncPort);
1299 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMEDIAPORT, &pThis->IMediaPort);
1300 return NULL;
1301}
1302
1303
1304/* -=-=-=-=- driver interface -=-=-=-=- */
1305
1306static int drvdiskintTreeDestroy(PAVLRFOFFNODECORE pNode, void *pvUser)
1307{
1308 PDRVDISKSEGMENT pSeg = (PDRVDISKSEGMENT)pNode;
1309
1310 RTMemFree(pSeg->pbSeg);
1311 RTMemFree(pSeg);
1312 return VINF_SUCCESS;
1313}
1314
1315/**
1316 * @copydoc FNPDMDRVDESTRUCT
1317 */
1318static DECLCALLBACK(void) drvdiskintDestruct(PPDMDRVINS pDrvIns)
1319{
1320 PDRVDISKINTEGRITY pThis = PDMINS_2_DATA(pDrvIns, PDRVDISKINTEGRITY);
1321
1322 if (pThis->pTreeSegments)
1323 {
1324 RTAvlrFileOffsetDestroy(pThis->pTreeSegments, drvdiskintTreeDestroy, NULL);
1325 RTMemFree(pThis->pTreeSegments);
1326 }
1327
1328 if (pThis->fTraceRequests)
1329 {
1330 pThis->fRunning = false;
1331 RTSemEventSignal(pThis->SemEvent);
1332 RTSemEventDestroy(pThis->SemEvent);
1333 }
1334
1335 if (pThis->fCheckDoubleCompletion)
1336 {
1337 /* Free all requests */
1338 while (pThis->papIoReq[pThis->iEntry])
1339 {
1340 RTMemFree(pThis->papIoReq[pThis->iEntry]);
1341 pThis->papIoReq[pThis->iEntry] = NULL;
1342 pThis->iEntry = (pThis->iEntry+1) % pThis->cEntries;
1343 }
1344 }
1345
1346 if (pThis->hIoLogger)
1347 VDDbgIoLogDestroy(pThis->hIoLogger);
1348}
1349
1350/**
1351 * Construct a disk integrity driver instance.
1352 *
1353 * @copydoc FNPDMDRVCONSTRUCT
1354 */
1355static DECLCALLBACK(int) drvdiskintConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
1356{
1357 int rc = VINF_SUCCESS;
1358 PDRVDISKINTEGRITY pThis = PDMINS_2_DATA(pDrvIns, PDRVDISKINTEGRITY);
1359 LogFlow(("drvdiskintConstruct: iInstance=%d\n", pDrvIns->iInstance));
1360 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
1361
1362 /*
1363 * Validate configuration.
1364 */
1365 if (!CFGMR3AreValuesValid(pCfg, "CheckConsistency\0"
1366 "TraceRequests\0"
1367 "CheckIntervalMs\0"
1368 "ExpireIntervalMs\0"
1369 "CheckDoubleCompletions\0"
1370 "HistorySize\0"
1371 "IoLog\0"
1372 "PrepopulateRamDisk\0"
1373 "ReadAfterWrite\0"
1374 "RecordWriteBeforeCompletion\0"))
1375 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
1376
1377 rc = CFGMR3QueryBoolDef(pCfg, "CheckConsistency", &pThis->fCheckConsistency, false);
1378 AssertRC(rc);
1379 rc = CFGMR3QueryBoolDef(pCfg, "TraceRequests", &pThis->fTraceRequests, false);
1380 AssertRC(rc);
1381 rc = CFGMR3QueryU32Def(pCfg, "CheckIntervalMs", &pThis->uCheckIntervalMs, 5000); /* 5 seconds */
1382 AssertRC(rc);
1383 rc = CFGMR3QueryU32Def(pCfg, "ExpireIntervalMs", &pThis->uExpireIntervalMs, 20000); /* 20 seconds */
1384 AssertRC(rc);
1385 rc = CFGMR3QueryBoolDef(pCfg, "CheckDoubleCompletions", &pThis->fCheckDoubleCompletion, false);
1386 AssertRC(rc);
1387 rc = CFGMR3QueryU32Def(pCfg, "HistorySize", &pThis->cEntries, 512);
1388 AssertRC(rc);
1389 rc = CFGMR3QueryBoolDef(pCfg, "PrepopulateRamDisk", &pThis->fPrepopulateRamDisk, false);
1390 AssertRC(rc);
1391 rc = CFGMR3QueryBoolDef(pCfg, "ReadAfterWrite", &pThis->fReadAfterWrite, false);
1392 AssertRC(rc);
1393 rc = CFGMR3QueryBoolDef(pCfg, "RecordWriteBeforeCompletion", &pThis->fRecordWriteBeforeCompletion, false);
1394 AssertRC(rc);
1395
1396 char *pszIoLogFilename = NULL;
1397 rc = CFGMR3QueryStringAlloc(pCfg, "IoLog", &pszIoLogFilename);
1398 Assert(RT_SUCCESS(rc) || rc == VERR_CFGM_VALUE_NOT_FOUND);
1399
1400 /*
1401 * Initialize most of the data members.
1402 */
1403 pThis->pDrvIns = pDrvIns;
1404
1405 /* IBase. */
1406 pDrvIns->IBase.pfnQueryInterface = drvdiskintQueryInterface;
1407
1408 /* IMedia */
1409 pThis->IMedia.pfnRead = drvdiskintRead;
1410 pThis->IMedia.pfnWrite = drvdiskintWrite;
1411 pThis->IMedia.pfnFlush = drvdiskintFlush;
1412 pThis->IMedia.pfnGetSize = drvdiskintGetSize;
1413 pThis->IMedia.pfnIsReadOnly = drvdiskintIsReadOnly;
1414 pThis->IMedia.pfnBiosGetPCHSGeometry = drvdiskintBiosGetPCHSGeometry;
1415 pThis->IMedia.pfnBiosSetPCHSGeometry = drvdiskintBiosSetPCHSGeometry;
1416 pThis->IMedia.pfnBiosGetLCHSGeometry = drvdiskintBiosGetLCHSGeometry;
1417 pThis->IMedia.pfnBiosSetLCHSGeometry = drvdiskintBiosSetLCHSGeometry;
1418 pThis->IMedia.pfnGetUuid = drvdiskintGetUuid;
1419 pThis->IMedia.pfnGetSectorSize = drvdiskintGetSectorSize;
1420 pThis->IMedia.pfnIoBufAlloc = drvdiskintIoBufAlloc;
1421 pThis->IMedia.pfnIoBufFree = drvdiskintIoBufFree;
1422 pThis->IMedia.pfnReadPcBios = drvdiskintReadPcBios;
1423
1424 /* IMediaAsync */
1425 pThis->IMediaAsync.pfnStartRead = drvdiskintStartRead;
1426 pThis->IMediaAsync.pfnStartWrite = drvdiskintStartWrite;
1427 pThis->IMediaAsync.pfnStartFlush = drvdiskintStartFlush;
1428
1429 /* IMediaAsyncPort. */
1430 pThis->IMediaAsyncPort.pfnTransferCompleteNotify = drvdiskintAsyncTransferCompleteNotify;
1431
1432 /* IMediaPort. */
1433 pThis->IMediaPort.pfnQueryDeviceLocation = drvdiskintQueryDeviceLocation;
1434
1435 /* Query the media port interface above us. */
1436 pThis->pDrvMediaPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIMEDIAPORT);
1437 if (!pThis->pDrvMediaPort)
1438 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_MISSING_INTERFACE_BELOW,
1439 N_("No media port inrerface above"));
1440
1441 /* Try to attach async media port interface above.*/
1442 pThis->pDrvMediaAsyncPort = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMIMEDIAASYNCPORT);
1443
1444 /*
1445 * Try attach driver below and query it's media interface.
1446 */
1447 PPDMIBASE pBase;
1448 rc = PDMDrvHlpAttach(pDrvIns, fFlags, &pBase);
1449 if (RT_FAILURE(rc))
1450 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1451 N_("Failed to attach driver below us! %Rrc"), rc);
1452
1453 pThis->pDrvMedia = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMEDIA);
1454 if (!pThis->pDrvMedia)
1455 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_MISSING_INTERFACE_BELOW,
1456 N_("No media or async media interface below"));
1457
1458 pThis->pDrvMediaAsync = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMEDIAASYNC);
1459
1460 if (pThis->pDrvMedia->pfnDiscard)
1461 pThis->IMedia.pfnDiscard = drvdiskintDiscard;
1462 if ( pThis->pDrvMediaAsync
1463 && pThis->pDrvMediaAsync->pfnStartDiscard)
1464 pThis->IMediaAsync.pfnStartDiscard = drvdiskintStartDiscard;
1465
1466 if (pThis->fCheckConsistency)
1467 {
1468 /* Create the AVL tree. */
1469 pThis->pTreeSegments = (PAVLRFOFFTREE)RTMemAllocZ(sizeof(AVLRFOFFTREE));
1470 if (!pThis->pTreeSegments)
1471 rc = VERR_NO_MEMORY;
1472 }
1473
1474 if (pThis->fTraceRequests)
1475 {
1476 for (unsigned i = 0; i < RT_ELEMENTS(pThis->apReqActive); i++)
1477 {
1478 pThis->apReqActive[i].pIoReq = NULL;
1479 pThis->apReqActive[i].tsStart = 0;
1480 }
1481
1482 pThis->iNextFreeSlot = 0;
1483
1484 /* Init event semaphore. */
1485 rc = RTSemEventCreate(&pThis->SemEvent);
1486 AssertRC(rc);
1487 pThis->fRunning = true;
1488 rc = RTThreadCreate(&pThis->hThread, drvdiskIntIoReqExpiredCheck, pThis,
1489 0, RTTHREADTYPE_INFREQUENT_POLLER, 0, "DiskIntegrity");
1490 AssertRC(rc);
1491 }
1492
1493 if (pThis->fCheckDoubleCompletion)
1494 {
1495 pThis->iEntry = 0;
1496 pThis->papIoReq = (PDRVDISKAIOREQ *)RTMemAllocZ(pThis->cEntries * sizeof(PDRVDISKAIOREQ));
1497 AssertPtr(pThis->papIoReq);
1498 }
1499
1500 if (pszIoLogFilename)
1501 {
1502 rc = VDDbgIoLogCreate(&pThis->hIoLogger, pszIoLogFilename, VDDBG_IOLOG_LOG_DATA);
1503 MMR3HeapFree(pszIoLogFilename);
1504 }
1505
1506 /* Read in all data before the start if requested. */
1507 if (pThis->fPrepopulateRamDisk)
1508 {
1509 uint64_t cbDisk = 0;
1510
1511 LogRel(("DiskIntegrity: Prepopulating RAM disk, this will take some time...\n"));
1512
1513 cbDisk = pThis->pDrvMedia->pfnGetSize(pThis->pDrvMedia);
1514 if (cbDisk)
1515 {
1516 uint64_t off = 0;
1517 uint8_t abBuffer[_64K];
1518 RTSGSEG Seg;
1519
1520 Seg.pvSeg = abBuffer;
1521
1522 while (cbDisk)
1523 {
1524 size_t cbThisRead = RT_MIN(cbDisk, sizeof(abBuffer));
1525
1526 rc = pThis->pDrvMedia->pfnRead(pThis->pDrvMedia, off, abBuffer, cbThisRead);
1527 if (RT_FAILURE(rc))
1528 break;
1529
1530 if (ASMBitFirstSet(abBuffer, sizeof(abBuffer) * 8) != -1)
1531 {
1532 Seg.cbSeg = cbThisRead;
1533 rc = drvdiskintWriteRecord(pThis, &Seg, 1,
1534 off, cbThisRead);
1535 if (RT_FAILURE(rc))
1536 break;
1537 }
1538
1539 cbDisk -= cbThisRead;
1540 off += cbThisRead;
1541 }
1542
1543 LogRel(("DiskIntegrity: Prepopulating RAM disk finished with %Rrc\n", rc));
1544 }
1545 else
1546 return PDMDRV_SET_ERROR(pDrvIns, VERR_INTERNAL_ERROR,
1547 N_("DiskIntegrity: Error querying the media size below"));
1548 }
1549
1550 return rc;
1551}
1552
1553
1554/**
1555 * Block driver registration record.
1556 */
1557const PDMDRVREG g_DrvDiskIntegrity =
1558{
1559 /* u32Version */
1560 PDM_DRVREG_VERSION,
1561 /* szName */
1562 "DiskIntegrity",
1563 /* szRCMod */
1564 "",
1565 /* szR0Mod */
1566 "",
1567 /* pszDescription */
1568 "Disk integrity driver.",
1569 /* fFlags */
1570 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1571 /* fClass. */
1572 PDM_DRVREG_CLASS_BLOCK,
1573 /* cMaxInstances */
1574 ~0U,
1575 /* cbInstance */
1576 sizeof(DRVDISKINTEGRITY),
1577 /* pfnConstruct */
1578 drvdiskintConstruct,
1579 /* pfnDestruct */
1580 drvdiskintDestruct,
1581 /* pfnRelocate */
1582 NULL,
1583 /* pfnIOCtl */
1584 NULL,
1585 /* pfnPowerOn */
1586 NULL,
1587 /* pfnReset */
1588 NULL,
1589 /* pfnSuspend */
1590 NULL,
1591 /* pfnResume */
1592 NULL,
1593 /* pfnAttach */
1594 NULL,
1595 /* pfnDetach */
1596 NULL,
1597 /* pfnPowerOff */
1598 NULL,
1599 /* pfnSoftReset */
1600 NULL,
1601 /* u32EndVersion */
1602 PDM_DRVREG_VERSION
1603};
1604
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