VirtualBox

source: vbox/trunk/src/VBox/Storage/VD.cpp@ 38449

Last change on this file since 38449 was 38449, checked in by vboxsync, 14 years ago

VD: Fix copy optimization

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 304.1 KB
Line 
1/* $Id: VD.cpp 38449 2011-08-14 10:53:13Z vboxsync $ */
2/** @file
3 * VBoxHDD - VBox HDD Container implementation.
4 */
5
6/*
7 * Copyright (C) 2006-2010 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* Header Files *
20*******************************************************************************/
21#define LOG_GROUP LOG_GROUP_VD
22#include <VBox/vd.h>
23#include <VBox/err.h>
24#include <VBox/sup.h>
25#include <VBox/log.h>
26
27#include <iprt/alloc.h>
28#include <iprt/assert.h>
29#include <iprt/uuid.h>
30#include <iprt/file.h>
31#include <iprt/string.h>
32#include <iprt/asm.h>
33#include <iprt/ldr.h>
34#include <iprt/dir.h>
35#include <iprt/path.h>
36#include <iprt/param.h>
37#include <iprt/memcache.h>
38#include <iprt/sg.h>
39#include <iprt/critsect.h>
40#include <iprt/list.h>
41#include <iprt/avl.h>
42
43#include <VBox/vd-plugin.h>
44#include <VBox/vd-cache-plugin.h>
45
46#define VBOXHDDDISK_SIGNATURE 0x6f0e2a7d
47
48/** Buffer size used for merging images. */
49#define VD_MERGE_BUFFER_SIZE (16 * _1M)
50
51/** Maximum number of segments in one I/O task. */
52#define VD_IO_TASK_SEGMENTS_MAX 64
53
54/**
55 * VD async I/O interface storage descriptor.
56 */
57typedef struct VDIIOFALLBACKSTORAGE
58{
59 /** File handle. */
60 RTFILE File;
61 /** Completion callback. */
62 PFNVDCOMPLETED pfnCompleted;
63 /** Thread for async access. */
64 RTTHREAD ThreadAsync;
65} VDIIOFALLBACKSTORAGE, *PVDIIOFALLBACKSTORAGE;
66
67/**
68 * Structure containing everything I/O related
69 * for the image and cache descriptors.
70 */
71typedef struct VDIO
72{
73 /** I/O interface to the upper layer. */
74 PVDINTERFACE pInterfaceIO;
75 /** I/O interface callback table. */
76 PVDINTERFACEIO pInterfaceIOCallbacks;
77
78 /** Per image internal I/O interface. */
79 VDINTERFACE VDIIOInt;
80
81 /** Fallback I/O interface, only used if the caller doesn't provide it. */
82 VDINTERFACE VDIIO;
83
84 /** Opaque backend data. */
85 void *pBackendData;
86 /** Disk this image is part of */
87 PVBOXHDD pDisk;
88} VDIO, *PVDIO;
89
90/**
91 * VBox HDD Container image descriptor.
92 */
93typedef struct VDIMAGE
94{
95 /** Link to parent image descriptor, if any. */
96 struct VDIMAGE *pPrev;
97 /** Link to child image descriptor, if any. */
98 struct VDIMAGE *pNext;
99 /** Container base filename. (UTF-8) */
100 char *pszFilename;
101 /** Data managed by the backend which keeps the actual info. */
102 void *pBackendData;
103 /** Cached sanitized image flags. */
104 unsigned uImageFlags;
105 /** Image open flags (only those handled generically in this code and which
106 * the backends will never ever see). */
107 unsigned uOpenFlags;
108
109 /** Function pointers for the various backend methods. */
110 PCVBOXHDDBACKEND Backend;
111 /** Pointer to list of VD interfaces, per-image. */
112 PVDINTERFACE pVDIfsImage;
113 /** I/O related things. */
114 VDIO VDIo;
115} VDIMAGE, *PVDIMAGE;
116
117/**
118 * uModified bit flags.
119 */
120#define VD_IMAGE_MODIFIED_FLAG RT_BIT(0)
121#define VD_IMAGE_MODIFIED_FIRST RT_BIT(1)
122#define VD_IMAGE_MODIFIED_DISABLE_UUID_UPDATE RT_BIT(2)
123
124
125/**
126 * VBox HDD Cache image descriptor.
127 */
128typedef struct VDCACHE
129{
130 /** Cache base filename. (UTF-8) */
131 char *pszFilename;
132 /** Data managed by the backend which keeps the actual info. */
133 void *pBackendData;
134 /** Cached sanitized image flags. */
135 unsigned uImageFlags;
136 /** Image open flags (only those handled generically in this code and which
137 * the backends will never ever see). */
138 unsigned uOpenFlags;
139
140 /** Function pointers for the various backend methods. */
141 PCVDCACHEBACKEND Backend;
142
143 /** Pointer to list of VD interfaces, per-cache. */
144 PVDINTERFACE pVDIfsCache;
145 /** I/O related things. */
146 VDIO VDIo;
147} VDCACHE, *PVDCACHE;
148
149/**
150 * VBox HDD Container main structure, private part.
151 */
152struct VBOXHDD
153{
154 /** Structure signature (VBOXHDDDISK_SIGNATURE). */
155 uint32_t u32Signature;
156
157 /** Image type. */
158 VDTYPE enmType;
159
160 /** Number of opened images. */
161 unsigned cImages;
162
163 /** Base image. */
164 PVDIMAGE pBase;
165
166 /** Last opened image in the chain.
167 * The same as pBase if only one image is used. */
168 PVDIMAGE pLast;
169
170 /** If a merge to one of the parents is running this may be non-NULL
171 * to indicate to what image the writes should be additionally relayed. */
172 PVDIMAGE pImageRelay;
173
174 /** Flags representing the modification state. */
175 unsigned uModified;
176
177 /** Cached size of this disk. */
178 uint64_t cbSize;
179 /** Cached PCHS geometry for this disk. */
180 VDGEOMETRY PCHSGeometry;
181 /** Cached LCHS geometry for this disk. */
182 VDGEOMETRY LCHSGeometry;
183
184 /** Pointer to list of VD interfaces, per-disk. */
185 PVDINTERFACE pVDIfsDisk;
186 /** Pointer to the common interface structure for error reporting. */
187 PVDINTERFACE pInterfaceError;
188 /** Pointer to the error interface callbacks we use if available. */
189 PVDINTERFACEERROR pInterfaceErrorCallbacks;
190
191 /** Pointer to the optional thread synchronization interface. */
192 PVDINTERFACE pInterfaceThreadSync;
193 /** Pointer to the optional thread synchronization callbacks. */
194 PVDINTERFACETHREADSYNC pInterfaceThreadSyncCallbacks;
195
196 /** Internal I/O interface callback table for the images. */
197 VDINTERFACEIOINT VDIIOIntCallbacks;
198
199 /** Callback table for the fallback I/O interface. */
200 VDINTERFACEIO VDIIOCallbacks;
201
202 /** Memory cache for I/O contexts */
203 RTMEMCACHE hMemCacheIoCtx;
204 /** Memory cache for I/O tasks. */
205 RTMEMCACHE hMemCacheIoTask;
206 /** Critical section protecting the disk against concurrent access. */
207 RTCRITSECT CritSect;
208 /** Flag whether the disk is currently locked by growing write or a flush
209 * request. Other flush or growing write requests need to wait until
210 * the current one completes.
211 */
212 volatile bool fLocked;
213 /** List of waiting requests. - Protected by the critical section. */
214 RTLISTNODE ListWriteLocked;
215 /** I/O context which locked the disk. */
216 PVDIOCTX pIoCtxLockOwner;
217
218 /** Pointer to the L2 disk cache if any. */
219 PVDCACHE pCache;
220};
221
222# define VD_THREAD_IS_CRITSECT_OWNER(Disk) \
223 do \
224 { \
225 AssertMsg(RTCritSectIsOwner(&Disk->CritSect), \
226 ("Thread does not own critical section\n"));\
227 } while(0)
228
229/**
230 * VBox parent read descriptor, used internally for compaction.
231 */
232typedef struct VDPARENTSTATEDESC
233{
234 /** Pointer to disk descriptor. */
235 PVBOXHDD pDisk;
236 /** Pointer to image descriptor. */
237 PVDIMAGE pImage;
238} VDPARENTSTATEDESC, *PVDPARENTSTATEDESC;
239
240/**
241 * Transfer direction.
242 */
243typedef enum VDIOCTXTXDIR
244{
245 /** Read */
246 VDIOCTXTXDIR_READ = 0,
247 /** Write */
248 VDIOCTXTXDIR_WRITE,
249 /** Flush */
250 VDIOCTXTXDIR_FLUSH,
251 /** 32bit hack */
252 VDIOCTXTXDIR_32BIT_HACK = 0x7fffffff
253} VDIOCTXTXDIR, *PVDIOCTXTXDIR;
254
255/** Transfer function */
256typedef DECLCALLBACK(int) FNVDIOCTXTRANSFER (PVDIOCTX pIoCtx);
257/** Pointer to a transfer function. */
258typedef FNVDIOCTXTRANSFER *PFNVDIOCTXTRANSFER;
259
260/**
261 * I/O context
262 */
263typedef struct VDIOCTX
264{
265 /** Disk this is request is for. */
266 PVBOXHDD pDisk;
267 /** Return code. */
268 int rcReq;
269 /** Transfer direction */
270 VDIOCTXTXDIR enmTxDir;
271 /** Number of bytes left until this context completes. */
272 volatile uint32_t cbTransferLeft;
273 /** Current offset */
274 volatile uint64_t uOffset;
275 /** Number of bytes to transfer */
276 volatile size_t cbTransfer;
277 /** Current image in the chain. */
278 PVDIMAGE pImageCur;
279 /** Start image to read from. pImageCur is reset to this
280 * value after it reached the first image in the chain. */
281 PVDIMAGE pImageStart;
282 /** S/G buffer */
283 RTSGBUF SgBuf;
284 /** Flag whether the I/O context is blocked because it is in the growing list. */
285 bool fBlocked;
286 /** Number of data transfers currently pending. */
287 volatile uint32_t cDataTransfersPending;
288 /** How many meta data transfers are pending. */
289 volatile uint32_t cMetaTransfersPending;
290 /** Flag whether the request finished */
291 volatile bool fComplete;
292 /** Temporary allocated memory which is freed
293 * when the context completes. */
294 void *pvAllocation;
295 /** Transfer function. */
296 PFNVDIOCTXTRANSFER pfnIoCtxTransfer;
297 /** Next transfer part after the current one completed. */
298 PFNVDIOCTXTRANSFER pfnIoCtxTransferNext;
299 /** Parent I/O context if any. Sets the type of the context (root/child) */
300 PVDIOCTX pIoCtxParent;
301 /** Type dependent data (root/child) */
302 union
303 {
304 /** Root data */
305 struct
306 {
307 /** Completion callback */
308 PFNVDASYNCTRANSFERCOMPLETE pfnComplete;
309 /** User argument 1 passed on completion. */
310 void *pvUser1;
311 /** User argument 1 passed on completion. */
312 void *pvUser2;
313 } Root;
314 /** Child data */
315 struct
316 {
317 /** Saved start offset */
318 uint64_t uOffsetSaved;
319 /** Saved transfer size */
320 size_t cbTransferLeftSaved;
321 /** Number of bytes transferred from the parent if this context completes. */
322 size_t cbTransferParent;
323 /** Number of bytes to pre read */
324 size_t cbPreRead;
325 /** Number of bytes to post read. */
326 size_t cbPostRead;
327 /** Number of bytes to write left in the parent. */
328 size_t cbWriteParent;
329 /** Write type dependent data. */
330 union
331 {
332 /** Optimized */
333 struct
334 {
335 /** Bytes to fill to satisfy the block size. Not part of the virtual disk. */
336 size_t cbFill;
337 /** Bytes to copy instead of reading from the parent */
338 size_t cbWriteCopy;
339 /** Bytes to read from the image. */
340 size_t cbReadImage;
341 } Optimized;
342 } Write;
343 } Child;
344 } Type;
345} VDIOCTX;
346
347typedef struct VDIOCTXDEFERRED
348{
349 /** Node in the list of deferred requests.
350 * A request can be deferred if the image is growing
351 * and the request accesses the same range or if
352 * the backend needs to read or write metadata from the disk
353 * before it can continue. */
354 RTLISTNODE NodeDeferred;
355 /** I/O context this entry points to. */
356 PVDIOCTX pIoCtx;
357} VDIOCTXDEFERRED, *PVDIOCTXDEFERRED;
358
359/**
360 * I/O task.
361 */
362typedef struct VDIOTASK
363{
364 /** Storage this task belongs to. */
365 PVDIOSTORAGE pIoStorage;
366 /** Optional completion callback. */
367 PFNVDXFERCOMPLETED pfnComplete;
368 /** Opaque user data. */
369 void *pvUser;
370 /** Flag whether this is a meta data transfer. */
371 bool fMeta;
372 /** Type dependent data. */
373 union
374 {
375 /** User data transfer. */
376 struct
377 {
378 /** Number of bytes this task transferred. */
379 uint32_t cbTransfer;
380 /** Pointer to the I/O context the task belongs. */
381 PVDIOCTX pIoCtx;
382 } User;
383 /** Meta data transfer. */
384 struct
385 {
386 /** Meta transfer this task is for. */
387 PVDMETAXFER pMetaXfer;
388 } Meta;
389 } Type;
390} VDIOTASK, *PVDIOTASK;
391
392/**
393 * Storage handle.
394 */
395typedef struct VDIOSTORAGE
396{
397 /** Image I/O state this storage handle belongs to. */
398 PVDIO pVDIo;
399 /** AVL tree for pending async metadata transfers. */
400 PAVLRFOFFTREE pTreeMetaXfers;
401 /** Storage handle */
402 void *pStorage;
403} VDIOSTORAGE;
404
405/**
406 * Metadata transfer.
407 *
408 * @note This entry can't be freed if either the list is not empty or
409 * the reference counter is not 0.
410 * The assumption is that the backends don't need to read huge amounts of
411 * metadata to complete a transfer so the additional memory overhead should
412 * be relatively small.
413 */
414typedef struct VDMETAXFER
415{
416 /** AVL core for fast search (the file offset is the key) */
417 AVLRFOFFNODECORE Core;
418 /** I/O storage for this transfer. */
419 PVDIOSTORAGE pIoStorage;
420 /** Flags. */
421 uint32_t fFlags;
422 /** List of I/O contexts waiting for this metadata transfer to complete. */
423 RTLISTNODE ListIoCtxWaiting;
424 /** Number of references to this entry. */
425 unsigned cRefs;
426 /** Size of the data stored with this entry. */
427 size_t cbMeta;
428 /** Data stored - variable size. */
429 uint8_t abData[1];
430} VDMETAXFER;
431
432/**
433 * The transfer direction for the metadata.
434 */
435#define VDMETAXFER_TXDIR_MASK 0x3
436#define VDMETAXFER_TXDIR_NONE 0x0
437#define VDMETAXFER_TXDIR_WRITE 0x1
438#define VDMETAXFER_TXDIR_READ 0x2
439#define VDMETAXFER_TXDIR_FLUSH 0x3
440#define VDMETAXFER_TXDIR_GET(flags) ((flags) & VDMETAXFER_TXDIR_MASK)
441#define VDMETAXFER_TXDIR_SET(flags, dir) ((flags) = (flags & ~VDMETAXFER_TXDIR_MASK) | (dir))
442
443extern VBOXHDDBACKEND g_RawBackend;
444extern VBOXHDDBACKEND g_VmdkBackend;
445extern VBOXHDDBACKEND g_VDIBackend;
446extern VBOXHDDBACKEND g_VhdBackend;
447extern VBOXHDDBACKEND g_ParallelsBackend;
448extern VBOXHDDBACKEND g_DmgBackend;
449extern VBOXHDDBACKEND g_ISCSIBackend;
450
451static unsigned g_cBackends = 0;
452static PVBOXHDDBACKEND *g_apBackends = NULL;
453static PVBOXHDDBACKEND aStaticBackends[] =
454{
455 &g_VmdkBackend,
456 &g_VDIBackend,
457 &g_VhdBackend,
458 &g_ParallelsBackend,
459 &g_DmgBackend,
460 &g_RawBackend,
461 &g_ISCSIBackend
462};
463
464/**
465 * Supported backends for the disk cache.
466 */
467extern VDCACHEBACKEND g_VciCacheBackend;
468
469static unsigned g_cCacheBackends = 0;
470static PVDCACHEBACKEND *g_apCacheBackends = NULL;
471static PVDCACHEBACKEND aStaticCacheBackends[] =
472{
473 &g_VciCacheBackend
474};
475
476/**
477 * internal: add several backends.
478 */
479static int vdAddBackends(PVBOXHDDBACKEND *ppBackends, unsigned cBackends)
480{
481 PVBOXHDDBACKEND *pTmp = (PVBOXHDDBACKEND*)RTMemRealloc(g_apBackends,
482 (g_cBackends + cBackends) * sizeof(PVBOXHDDBACKEND));
483 if (RT_UNLIKELY(!pTmp))
484 return VERR_NO_MEMORY;
485 g_apBackends = pTmp;
486 memcpy(&g_apBackends[g_cBackends], ppBackends, cBackends * sizeof(PVBOXHDDBACKEND));
487 g_cBackends += cBackends;
488 return VINF_SUCCESS;
489}
490
491/**
492 * internal: add single backend.
493 */
494DECLINLINE(int) vdAddBackend(PVBOXHDDBACKEND pBackend)
495{
496 return vdAddBackends(&pBackend, 1);
497}
498
499/**
500 * internal: add several cache backends.
501 */
502static int vdAddCacheBackends(PVDCACHEBACKEND *ppBackends, unsigned cBackends)
503{
504 PVDCACHEBACKEND *pTmp = (PVDCACHEBACKEND*)RTMemRealloc(g_apCacheBackends,
505 (g_cCacheBackends + cBackends) * sizeof(PVDCACHEBACKEND));
506 if (RT_UNLIKELY(!pTmp))
507 return VERR_NO_MEMORY;
508 g_apCacheBackends = pTmp;
509 memcpy(&g_apCacheBackends[g_cCacheBackends], ppBackends, cBackends * sizeof(PVDCACHEBACKEND));
510 g_cCacheBackends += cBackends;
511 return VINF_SUCCESS;
512}
513
514/**
515 * internal: add single cache backend.
516 */
517DECLINLINE(int) vdAddCacheBackend(PVDCACHEBACKEND pBackend)
518{
519 return vdAddCacheBackends(&pBackend, 1);
520}
521
522/**
523 * internal: issue error message.
524 */
525static int vdError(PVBOXHDD pDisk, int rc, RT_SRC_POS_DECL,
526 const char *pszFormat, ...)
527{
528 va_list va;
529 va_start(va, pszFormat);
530 if (pDisk->pInterfaceErrorCallbacks)
531 pDisk->pInterfaceErrorCallbacks->pfnError(pDisk->pInterfaceError->pvUser, rc, RT_SRC_POS_ARGS, pszFormat, va);
532 va_end(va);
533 return rc;
534}
535
536/**
537 * internal: thread synchronization, start read.
538 */
539DECLINLINE(int) vdThreadStartRead(PVBOXHDD pDisk)
540{
541 int rc = VINF_SUCCESS;
542 if (RT_UNLIKELY(pDisk->pInterfaceThreadSyncCallbacks))
543 rc = pDisk->pInterfaceThreadSyncCallbacks->pfnStartRead(pDisk->pInterfaceThreadSync->pvUser);
544 return rc;
545}
546
547/**
548 * internal: thread synchronization, finish read.
549 */
550DECLINLINE(int) vdThreadFinishRead(PVBOXHDD pDisk)
551{
552 int rc = VINF_SUCCESS;
553 if (RT_UNLIKELY(pDisk->pInterfaceThreadSyncCallbacks))
554 rc = pDisk->pInterfaceThreadSyncCallbacks->pfnFinishRead(pDisk->pInterfaceThreadSync->pvUser);
555 return rc;
556}
557
558/**
559 * internal: thread synchronization, start write.
560 */
561DECLINLINE(int) vdThreadStartWrite(PVBOXHDD pDisk)
562{
563 int rc = VINF_SUCCESS;
564 if (RT_UNLIKELY(pDisk->pInterfaceThreadSyncCallbacks))
565 rc = pDisk->pInterfaceThreadSyncCallbacks->pfnStartWrite(pDisk->pInterfaceThreadSync->pvUser);
566 return rc;
567}
568
569/**
570 * internal: thread synchronization, finish write.
571 */
572DECLINLINE(int) vdThreadFinishWrite(PVBOXHDD pDisk)
573{
574 int rc = VINF_SUCCESS;
575 if (RT_UNLIKELY(pDisk->pInterfaceThreadSyncCallbacks))
576 rc = pDisk->pInterfaceThreadSyncCallbacks->pfnFinishWrite(pDisk->pInterfaceThreadSync->pvUser);
577 return rc;
578}
579
580/**
581 * internal: find image format backend.
582 */
583static int vdFindBackend(const char *pszBackend, PCVBOXHDDBACKEND *ppBackend)
584{
585 int rc = VINF_SUCCESS;
586 PCVBOXHDDBACKEND pBackend = NULL;
587
588 if (!g_apBackends)
589 VDInit();
590
591 for (unsigned i = 0; i < g_cBackends; i++)
592 {
593 if (!RTStrICmp(pszBackend, g_apBackends[i]->pszBackendName))
594 {
595 pBackend = g_apBackends[i];
596 break;
597 }
598 }
599 *ppBackend = pBackend;
600 return rc;
601}
602
603/**
604 * internal: find cache format backend.
605 */
606static int vdFindCacheBackend(const char *pszBackend, PCVDCACHEBACKEND *ppBackend)
607{
608 int rc = VINF_SUCCESS;
609 PCVDCACHEBACKEND pBackend = NULL;
610
611 if (!g_apCacheBackends)
612 VDInit();
613
614 for (unsigned i = 0; i < g_cCacheBackends; i++)
615 {
616 if (!RTStrICmp(pszBackend, g_apCacheBackends[i]->pszBackendName))
617 {
618 pBackend = g_apCacheBackends[i];
619 break;
620 }
621 }
622 *ppBackend = pBackend;
623 return rc;
624}
625
626/**
627 * internal: add image structure to the end of images list.
628 */
629static void vdAddImageToList(PVBOXHDD pDisk, PVDIMAGE pImage)
630{
631 pImage->pPrev = NULL;
632 pImage->pNext = NULL;
633
634 if (pDisk->pBase)
635 {
636 Assert(pDisk->cImages > 0);
637 pImage->pPrev = pDisk->pLast;
638 pDisk->pLast->pNext = pImage;
639 pDisk->pLast = pImage;
640 }
641 else
642 {
643 Assert(pDisk->cImages == 0);
644 pDisk->pBase = pImage;
645 pDisk->pLast = pImage;
646 }
647
648 pDisk->cImages++;
649}
650
651/**
652 * internal: remove image structure from the images list.
653 */
654static void vdRemoveImageFromList(PVBOXHDD pDisk, PVDIMAGE pImage)
655{
656 Assert(pDisk->cImages > 0);
657
658 if (pImage->pPrev)
659 pImage->pPrev->pNext = pImage->pNext;
660 else
661 pDisk->pBase = pImage->pNext;
662
663 if (pImage->pNext)
664 pImage->pNext->pPrev = pImage->pPrev;
665 else
666 pDisk->pLast = pImage->pPrev;
667
668 pImage->pPrev = NULL;
669 pImage->pNext = NULL;
670
671 pDisk->cImages--;
672}
673
674/**
675 * internal: find image by index into the images list.
676 */
677static PVDIMAGE vdGetImageByNumber(PVBOXHDD pDisk, unsigned nImage)
678{
679 PVDIMAGE pImage = pDisk->pBase;
680 if (nImage == VD_LAST_IMAGE)
681 return pDisk->pLast;
682 while (pImage && nImage)
683 {
684 pImage = pImage->pNext;
685 nImage--;
686 }
687 return pImage;
688}
689
690/**
691 * Internal: Tries to read the desired range from the given cache.
692 *
693 * @returns VBox status code.
694 * @retval VERR_VD_BLOCK_FREE if the block is not in the cache.
695 * pcbRead will be set to the number of bytes not in the cache.
696 * Everything thereafter might be in the cache.
697 * @param pCache The cache to read from.
698 * @param uOffset Offset of the virtual disk to read.
699 * @param pvBuf Where to store the read data.
700 * @param cbRead How much to read.
701 * @param pcbRead Where to store the number of bytes actually read.
702 * On success this indicates the number of bytes read from the cache.
703 * If VERR_VD_BLOCK_FREE is returned this gives the number of bytes
704 * which are not in the cache.
705 * In both cases everything beyond this value
706 * might or might not be in the cache.
707 */
708static int vdCacheReadHelper(PVDCACHE pCache, uint64_t uOffset,
709 void *pvBuf, size_t cbRead, size_t *pcbRead)
710{
711 int rc = VINF_SUCCESS;
712
713 LogFlowFunc(("pCache=%#p uOffset=%llu pvBuf=%#p cbRead=%zu pcbRead=%#p\n",
714 pCache, uOffset, pvBuf, cbRead, pcbRead));
715
716 AssertPtr(pCache);
717 AssertPtr(pcbRead);
718
719 rc = pCache->Backend->pfnRead(pCache->pBackendData, uOffset, pvBuf,
720 cbRead, pcbRead);
721
722 LogFlowFunc(("returns rc=%Rrc pcbRead=%zu\n", rc, *pcbRead));
723 return rc;
724}
725
726/**
727 * Internal: Writes data for the given block into the cache.
728 *
729 * @returns VBox status code.
730 * @param pCache The cache to write to.
731 * @param uOffset Offset of the virtual disk to write to teh cache.
732 * @param pcvBuf The data to write.
733 * @param cbWrite How much to write.
734 * @param pcbWritten How much data could be written, optional.
735 */
736static int vdCacheWriteHelper(PVDCACHE pCache, uint64_t uOffset, const void *pcvBuf,
737 size_t cbWrite, size_t *pcbWritten)
738{
739 int rc = VINF_SUCCESS;
740
741 LogFlowFunc(("pCache=%#p uOffset=%llu pvBuf=%#p cbWrite=%zu pcbWritten=%#p\n",
742 pCache, uOffset, pcvBuf, cbWrite, pcbWritten));
743
744 AssertPtr(pCache);
745 AssertPtr(pcvBuf);
746 Assert(cbWrite > 0);
747
748 if (pcbWritten)
749 rc = pCache->Backend->pfnWrite(pCache->pBackendData, uOffset, pcvBuf,
750 cbWrite, pcbWritten);
751 else
752 {
753 size_t cbWritten = 0;
754
755 do
756 {
757 rc = pCache->Backend->pfnWrite(pCache->pBackendData, uOffset, pcvBuf,
758 cbWrite, &cbWritten);
759 uOffset += cbWritten;
760 pcvBuf = (char *)pcvBuf + cbWritten;
761 cbWrite -= cbWritten;
762 } while ( cbWrite
763 && RT_SUCCESS(rc));
764 }
765
766 LogFlowFunc(("returns rc=%Rrc pcbWritten=%zu\n",
767 rc, pcbWritten ? *pcbWritten : cbWrite));
768 return rc;
769}
770
771/**
772 * Internal: Reads a given amount of data from the image chain of the disk.
773 **/
774static int vdDiskReadHelper(PVBOXHDD pDisk, PVDIMAGE pImage, PVDIMAGE pImageParentOverride,
775 uint64_t uOffset, void *pvBuf, size_t cbRead, size_t *pcbThisRead)
776{
777 int rc = VINF_SUCCESS;
778 size_t cbThisRead = cbRead;
779
780 AssertPtr(pcbThisRead);
781
782 *pcbThisRead = 0;
783
784 /*
785 * Try to read from the given image.
786 * If the block is not allocated read from override chain if present.
787 */
788 rc = pImage->Backend->pfnRead(pImage->pBackendData,
789 uOffset, pvBuf, cbThisRead,
790 &cbThisRead);
791
792 if (rc == VERR_VD_BLOCK_FREE)
793 {
794 for (PVDIMAGE pCurrImage = pImageParentOverride ? pImageParentOverride : pImage->pPrev;
795 pCurrImage != NULL && rc == VERR_VD_BLOCK_FREE;
796 pCurrImage = pCurrImage->pPrev)
797 {
798 rc = pCurrImage->Backend->pfnRead(pCurrImage->pBackendData,
799 uOffset, pvBuf, cbThisRead,
800 &cbThisRead);
801 }
802 }
803
804 if (RT_SUCCESS(rc) || rc == VERR_VD_BLOCK_FREE)
805 *pcbThisRead = cbThisRead;
806
807 return rc;
808}
809
810/**
811 * Extended version of vdReadHelper(), implementing certain optimizations
812 * for image cloning.
813 *
814 * @returns VBox status code.
815 * @param pDisk The disk to read from.
816 * @param pImage The image to start reading from.
817 * @param pImageParentOverride The parent image to read from
818 * if the starting image returns a free block.
819 * If NULL is passed the real parent of the image
820 * in the chain is used.
821 * @param uOffset Offset in the disk to start reading from.
822 * @param pvBuf Where to store the read data.
823 * @param cbRead How much to read.
824 * @param fZeroFreeBlocks Flag whether free blocks should be zeroed.
825 * If false and no image has data for sepcified
826 * range VERR_VD_BLOCK_FREE is returned.
827 * Note that unallocated blocks are still zeroed
828 * if at least one image has valid data for a part
829 * of the range.
830 * @param fUpdateCache Flag whether to update the attached cache if
831 * available.
832 * @param cImagesRead Number of images in the chain to read until
833 * the read is cut off. A value of 0 disables the cut off.
834 */
835static int vdReadHelperEx(PVBOXHDD pDisk, PVDIMAGE pImage, PVDIMAGE pImageParentOverride,
836 uint64_t uOffset, void *pvBuf, size_t cbRead,
837 bool fZeroFreeBlocks, bool fUpdateCache, unsigned cImagesRead)
838{
839 int rc = VINF_SUCCESS;
840 size_t cbThisRead;
841 bool fAllFree = true;
842 size_t cbBufClear = 0;
843
844 /* Loop until all read. */
845 do
846 {
847 /* Search for image with allocated block. Do not attempt to read more
848 * than the previous reads marked as valid. Otherwise this would return
849 * stale data when different block sizes are used for the images. */
850 cbThisRead = cbRead;
851
852 if ( pDisk->pCache
853 && !pImageParentOverride)
854 {
855 rc = vdCacheReadHelper(pDisk->pCache, uOffset, pvBuf,
856 cbThisRead, &cbThisRead);
857
858 if (rc == VERR_VD_BLOCK_FREE)
859 {
860 rc = vdDiskReadHelper(pDisk, pImage, NULL, uOffset, pvBuf, cbThisRead,
861 &cbThisRead);
862
863 /* If the read was successful, write the data back into the cache. */
864 if ( RT_SUCCESS(rc)
865 && fUpdateCache)
866 {
867 rc = vdCacheWriteHelper(pDisk->pCache, uOffset, pvBuf,
868 cbThisRead, NULL);
869 }
870 }
871 }
872 else
873 {
874 /** @todo can be be replaced by vdDiskReadHelper if it proves to be reliable,
875 * don't want to be responsible for data corruption...
876 */
877 /*
878 * Try to read from the given image.
879 * If the block is not allocated read from override chain if present.
880 */
881 rc = pImage->Backend->pfnRead(pImage->pBackendData,
882 uOffset, pvBuf, cbThisRead,
883 &cbThisRead);
884
885 if ( rc == VERR_VD_BLOCK_FREE
886 && cImagesRead != 1)
887 {
888 unsigned cImagesToProcess = cImagesRead;
889
890 for (PVDIMAGE pCurrImage = pImageParentOverride ? pImageParentOverride : pImage->pPrev;
891 pCurrImage != NULL && rc == VERR_VD_BLOCK_FREE;
892 pCurrImage = pCurrImage->pPrev)
893 {
894 rc = pCurrImage->Backend->pfnRead(pCurrImage->pBackendData,
895 uOffset, pvBuf, cbThisRead,
896 &cbThisRead);
897 if (cImagesToProcess == 1)
898 break;
899 else if (cImagesToProcess > 0)
900 cImagesToProcess--;
901 }
902 }
903 }
904
905 /* No image in the chain contains the data for the block. */
906 if (rc == VERR_VD_BLOCK_FREE)
907 {
908 /* Fill the free space with 0 if we are told to do so
909 * or a previous read returned valid data. */
910 if (fZeroFreeBlocks || !fAllFree)
911 memset(pvBuf, '\0', cbThisRead);
912 else
913 cbBufClear += cbThisRead;
914
915 rc = VINF_SUCCESS;
916 }
917 else if (RT_SUCCESS(rc))
918 {
919 /* First not free block, fill the space before with 0. */
920 if (!fZeroFreeBlocks)
921 {
922 memset((char *)pvBuf - cbBufClear, '\0', cbBufClear);
923 cbBufClear = 0;
924 fAllFree = false;
925 }
926 }
927
928 cbRead -= cbThisRead;
929 uOffset += cbThisRead;
930 pvBuf = (char *)pvBuf + cbThisRead;
931 } while (cbRead != 0 && RT_SUCCESS(rc));
932
933 return (!fZeroFreeBlocks && fAllFree) ? VERR_VD_BLOCK_FREE : rc;
934}
935
936/**
937 * internal: read the specified amount of data in whatever blocks the backend
938 * will give us.
939 */
940static int vdReadHelper(PVBOXHDD pDisk, PVDIMAGE pImage, uint64_t uOffset,
941 void *pvBuf, size_t cbRead, bool fUpdateCache)
942{
943 return vdReadHelperEx(pDisk, pImage, NULL, uOffset, pvBuf, cbRead,
944 true /* fZeroFreeBlocks */, fUpdateCache, 0);
945}
946
947DECLINLINE(PVDIOCTX) vdIoCtxAlloc(PVBOXHDD pDisk, VDIOCTXTXDIR enmTxDir,
948 uint64_t uOffset, size_t cbTransfer,
949 PVDIMAGE pImageStart,
950 PCRTSGBUF pcSgBuf, void *pvAllocation,
951 PFNVDIOCTXTRANSFER pfnIoCtxTransfer)
952{
953 PVDIOCTX pIoCtx = NULL;
954
955 pIoCtx = (PVDIOCTX)RTMemCacheAlloc(pDisk->hMemCacheIoCtx);
956 if (RT_LIKELY(pIoCtx))
957 {
958 pIoCtx->pDisk = pDisk;
959 pIoCtx->enmTxDir = enmTxDir;
960 pIoCtx->cbTransferLeft = cbTransfer;
961 pIoCtx->uOffset = uOffset;
962 pIoCtx->cbTransfer = cbTransfer;
963 pIoCtx->pImageStart = pImageStart;
964 pIoCtx->pImageCur = pImageStart;
965 pIoCtx->cDataTransfersPending = 0;
966 pIoCtx->cMetaTransfersPending = 0;
967 pIoCtx->fComplete = false;
968 pIoCtx->fBlocked = false;
969 pIoCtx->pvAllocation = pvAllocation;
970 pIoCtx->pfnIoCtxTransfer = pfnIoCtxTransfer;
971 pIoCtx->pfnIoCtxTransferNext = NULL;
972 pIoCtx->rcReq = VINF_SUCCESS;
973
974 /* There is no S/G list for a flush request. */
975 if (enmTxDir != VDIOCTXTXDIR_FLUSH)
976 RTSgBufClone(&pIoCtx->SgBuf, pcSgBuf);
977 else
978 memset(&pIoCtx->SgBuf, 0, sizeof(RTSGBUF));
979 }
980
981 return pIoCtx;
982}
983
984DECLINLINE(PVDIOCTX) vdIoCtxRootAlloc(PVBOXHDD pDisk, VDIOCTXTXDIR enmTxDir,
985 uint64_t uOffset, size_t cbTransfer,
986 PVDIMAGE pImageStart, PCRTSGBUF pcSgBuf,
987 PFNVDASYNCTRANSFERCOMPLETE pfnComplete,
988 void *pvUser1, void *pvUser2,
989 void *pvAllocation,
990 PFNVDIOCTXTRANSFER pfnIoCtxTransfer)
991{
992 PVDIOCTX pIoCtx = vdIoCtxAlloc(pDisk, enmTxDir, uOffset, cbTransfer, pImageStart,
993 pcSgBuf, pvAllocation, pfnIoCtxTransfer);
994
995 if (RT_LIKELY(pIoCtx))
996 {
997 pIoCtx->pIoCtxParent = NULL;
998 pIoCtx->Type.Root.pfnComplete = pfnComplete;
999 pIoCtx->Type.Root.pvUser1 = pvUser1;
1000 pIoCtx->Type.Root.pvUser2 = pvUser2;
1001 }
1002
1003 LogFlow(("Allocated root I/O context %#p\n", pIoCtx));
1004 return pIoCtx;
1005}
1006
1007DECLINLINE(PVDIOCTX) vdIoCtxChildAlloc(PVBOXHDD pDisk, VDIOCTXTXDIR enmTxDir,
1008 uint64_t uOffset, size_t cbTransfer,
1009 PVDIMAGE pImageStart, PCRTSGBUF pcSgBuf,
1010 PVDIOCTX pIoCtxParent, size_t cbTransferParent,
1011 size_t cbWriteParent, void *pvAllocation,
1012 PFNVDIOCTXTRANSFER pfnIoCtxTransfer)
1013{
1014 PVDIOCTX pIoCtx = vdIoCtxAlloc(pDisk, enmTxDir, uOffset, cbTransfer, pImageStart,
1015 pcSgBuf, pvAllocation, pfnIoCtxTransfer);
1016
1017 AssertPtr(pIoCtxParent);
1018 Assert(!pIoCtxParent->pIoCtxParent);
1019
1020 if (RT_LIKELY(pIoCtx))
1021 {
1022 pIoCtx->pIoCtxParent = pIoCtxParent;
1023 pIoCtx->Type.Child.uOffsetSaved = uOffset;
1024 pIoCtx->Type.Child.cbTransferLeftSaved = cbTransfer;
1025 pIoCtx->Type.Child.cbTransferParent = cbTransferParent;
1026 pIoCtx->Type.Child.cbWriteParent = cbWriteParent;
1027 }
1028
1029 LogFlow(("Allocated child I/O context %#p\n", pIoCtx));
1030 return pIoCtx;
1031}
1032
1033DECLINLINE(PVDIOTASK) vdIoTaskUserAlloc(PVDIOSTORAGE pIoStorage, PFNVDXFERCOMPLETED pfnComplete, void *pvUser, PVDIOCTX pIoCtx, uint32_t cbTransfer)
1034{
1035 PVDIOTASK pIoTask = NULL;
1036
1037 pIoTask = (PVDIOTASK)RTMemCacheAlloc(pIoStorage->pVDIo->pDisk->hMemCacheIoTask);
1038 if (pIoTask)
1039 {
1040 pIoTask->pIoStorage = pIoStorage;
1041 pIoTask->pfnComplete = pfnComplete;
1042 pIoTask->pvUser = pvUser;
1043 pIoTask->fMeta = false;
1044 pIoTask->Type.User.cbTransfer = cbTransfer;
1045 pIoTask->Type.User.pIoCtx = pIoCtx;
1046 }
1047
1048 return pIoTask;
1049}
1050
1051DECLINLINE(PVDIOTASK) vdIoTaskMetaAlloc(PVDIOSTORAGE pIoStorage, PFNVDXFERCOMPLETED pfnComplete, void *pvUser, PVDMETAXFER pMetaXfer)
1052{
1053 PVDIOTASK pIoTask = NULL;
1054
1055 pIoTask = (PVDIOTASK)RTMemCacheAlloc(pIoStorage->pVDIo->pDisk->hMemCacheIoTask);
1056 if (pIoTask)
1057 {
1058 pIoTask->pIoStorage = pIoStorage;
1059 pIoTask->pfnComplete = pfnComplete;
1060 pIoTask->pvUser = pvUser;
1061 pIoTask->fMeta = true;
1062 pIoTask->Type.Meta.pMetaXfer = pMetaXfer;
1063 }
1064
1065 return pIoTask;
1066}
1067
1068DECLINLINE(void) vdIoCtxFree(PVBOXHDD pDisk, PVDIOCTX pIoCtx)
1069{
1070 LogFlow(("Freeing I/O context %#p\n", pIoCtx));
1071 if (pIoCtx->pvAllocation)
1072 RTMemFree(pIoCtx->pvAllocation);
1073#ifdef DEBUG
1074 memset(pIoCtx, 0xff, sizeof(VDIOCTX));
1075#endif
1076 RTMemCacheFree(pDisk->hMemCacheIoCtx, pIoCtx);
1077}
1078
1079DECLINLINE(void) vdIoTaskFree(PVBOXHDD pDisk, PVDIOTASK pIoTask)
1080{
1081 RTMemCacheFree(pDisk->hMemCacheIoTask, pIoTask);
1082}
1083
1084DECLINLINE(void) vdIoCtxChildReset(PVDIOCTX pIoCtx)
1085{
1086 AssertPtr(pIoCtx->pIoCtxParent);
1087
1088 RTSgBufReset(&pIoCtx->SgBuf);
1089 pIoCtx->uOffset = pIoCtx->Type.Child.uOffsetSaved;
1090 pIoCtx->cbTransferLeft = pIoCtx->Type.Child.cbTransferLeftSaved;
1091}
1092
1093DECLINLINE(PVDMETAXFER) vdMetaXferAlloc(PVDIOSTORAGE pIoStorage, uint64_t uOffset, size_t cb)
1094{
1095 PVDMETAXFER pMetaXfer = (PVDMETAXFER)RTMemAlloc(RT_OFFSETOF(VDMETAXFER, abData[cb]));
1096
1097 if (RT_LIKELY(pMetaXfer))
1098 {
1099 pMetaXfer->Core.Key = uOffset;
1100 pMetaXfer->Core.KeyLast = uOffset + cb - 1;
1101 pMetaXfer->fFlags = VDMETAXFER_TXDIR_NONE;
1102 pMetaXfer->cbMeta = cb;
1103 pMetaXfer->pIoStorage = pIoStorage;
1104 pMetaXfer->cRefs = 0;
1105 RTListInit(&pMetaXfer->ListIoCtxWaiting);
1106 }
1107 return pMetaXfer;
1108}
1109
1110DECLINLINE(int) vdIoCtxDefer(PVBOXHDD pDisk, PVDIOCTX pIoCtx)
1111{
1112 PVDIOCTXDEFERRED pDeferred = (PVDIOCTXDEFERRED)RTMemAllocZ(sizeof(VDIOCTXDEFERRED));
1113
1114 if (!pDeferred)
1115 return VERR_NO_MEMORY;
1116
1117 LogFlowFunc(("Deferring write pIoCtx=%#p\n", pIoCtx));
1118
1119 Assert(!pIoCtx->pIoCtxParent && !pIoCtx->fBlocked);
1120
1121 RTListInit(&pDeferred->NodeDeferred);
1122 pDeferred->pIoCtx = pIoCtx;
1123 RTListAppend(&pDisk->ListWriteLocked, &pDeferred->NodeDeferred);
1124 pIoCtx->fBlocked = true;
1125 return VINF_SUCCESS;
1126}
1127
1128static size_t vdIoCtxCopy(PVDIOCTX pIoCtxDst, PVDIOCTX pIoCtxSrc, size_t cbData)
1129{
1130 return RTSgBufCopy(&pIoCtxDst->SgBuf, &pIoCtxSrc->SgBuf, cbData);
1131}
1132
1133static int vdIoCtxCmp(PVDIOCTX pIoCtx1, PVDIOCTX pIoCtx2, size_t cbData)
1134{
1135 return RTSgBufCmp(&pIoCtx1->SgBuf, &pIoCtx2->SgBuf, cbData);
1136}
1137
1138static size_t vdIoCtxCopyTo(PVDIOCTX pIoCtx, uint8_t *pbData, size_t cbData)
1139{
1140 return RTSgBufCopyToBuf(&pIoCtx->SgBuf, pbData, cbData);
1141}
1142
1143
1144static size_t vdIoCtxCopyFrom(PVDIOCTX pIoCtx, uint8_t *pbData, size_t cbData)
1145{
1146 return RTSgBufCopyFromBuf(&pIoCtx->SgBuf, pbData, cbData);
1147}
1148
1149static size_t vdIoCtxSet(PVDIOCTX pIoCtx, uint8_t ch, size_t cbData)
1150{
1151 return RTSgBufSet(&pIoCtx->SgBuf, ch, cbData);
1152}
1153
1154static int vdIoCtxProcess(PVDIOCTX pIoCtx)
1155{
1156 int rc = VINF_SUCCESS;
1157 PVBOXHDD pDisk = pIoCtx->pDisk;
1158
1159 LogFlowFunc(("pIoCtx=%#p\n", pIoCtx));
1160
1161 RTCritSectEnter(&pDisk->CritSect);
1162
1163 if ( !pIoCtx->cbTransferLeft
1164 && !pIoCtx->cMetaTransfersPending
1165 && !pIoCtx->cDataTransfersPending
1166 && !pIoCtx->pfnIoCtxTransfer)
1167 {
1168 rc = VINF_VD_ASYNC_IO_FINISHED;
1169 goto out;
1170 }
1171
1172 /*
1173 * We complete the I/O context in case of an error
1174 * if there is no I/O task pending.
1175 */
1176 if ( RT_FAILURE(pIoCtx->rcReq)
1177 && !pIoCtx->cMetaTransfersPending
1178 && !pIoCtx->cDataTransfersPending)
1179 {
1180 rc = VINF_VD_ASYNC_IO_FINISHED;
1181 goto out;
1182 }
1183
1184 /* Don't change anything if there is a metadata transfer pending or we are blocked. */
1185 if ( pIoCtx->cMetaTransfersPending
1186 || pIoCtx->fBlocked)
1187 {
1188 rc = VERR_VD_ASYNC_IO_IN_PROGRESS;
1189 goto out;
1190 }
1191
1192 if (pIoCtx->pfnIoCtxTransfer)
1193 {
1194 /* Call the transfer function advancing to the next while there is no error. */
1195 while ( pIoCtx->pfnIoCtxTransfer
1196 && RT_SUCCESS(rc))
1197 {
1198 LogFlowFunc(("calling transfer function %#p\n", pIoCtx->pfnIoCtxTransfer));
1199 rc = pIoCtx->pfnIoCtxTransfer(pIoCtx);
1200
1201 /* Advance to the next part of the transfer if the current one succeeded. */
1202 if (RT_SUCCESS(rc))
1203 {
1204 pIoCtx->pfnIoCtxTransfer = pIoCtx->pfnIoCtxTransferNext;
1205 pIoCtx->pfnIoCtxTransferNext = NULL;
1206 }
1207 }
1208 }
1209
1210 if ( RT_SUCCESS(rc)
1211 && !pIoCtx->cbTransferLeft
1212 && !pIoCtx->cMetaTransfersPending
1213 && !pIoCtx->cDataTransfersPending)
1214 rc = VINF_VD_ASYNC_IO_FINISHED;
1215 else if ( RT_SUCCESS(rc)
1216 || rc == VERR_VD_NOT_ENOUGH_METADATA
1217 || rc == VERR_VD_IOCTX_HALT)
1218 rc = VERR_VD_ASYNC_IO_IN_PROGRESS;
1219 else if (RT_FAILURE(rc) && (rc != VERR_VD_ASYNC_IO_IN_PROGRESS))
1220 {
1221 ASMAtomicCmpXchgS32(&pIoCtx->rcReq, rc, VINF_SUCCESS);
1222 /*
1223 * The I/O context completed if we have an error and there is no data
1224 * or meta data transfer pending.
1225 */
1226 if ( !pIoCtx->cMetaTransfersPending
1227 && !pIoCtx->cDataTransfersPending)
1228 rc = VINF_VD_ASYNC_IO_FINISHED;
1229 else
1230 rc = VERR_VD_ASYNC_IO_IN_PROGRESS;
1231 }
1232
1233out:
1234 RTCritSectLeave(&pDisk->CritSect);
1235
1236 LogFlowFunc(("pIoCtx=%#p rc=%Rrc cbTransferLeft=%u cMetaTransfersPending=%u fComplete=%RTbool\n",
1237 pIoCtx, rc, pIoCtx->cbTransferLeft, pIoCtx->cMetaTransfersPending,
1238 pIoCtx->fComplete));
1239
1240 return rc;
1241}
1242
1243DECLINLINE(bool) vdIoCtxIsDiskLockOwner(PVBOXHDD pDisk, PVDIOCTX pIoCtx)
1244{
1245 return pDisk->fLocked
1246 && pDisk->pIoCtxLockOwner == pIoCtx;
1247}
1248
1249static int vdIoCtxLockDisk(PVBOXHDD pDisk, PVDIOCTX pIoCtx)
1250{
1251 int rc = VINF_SUCCESS;
1252
1253 LogFlowFunc(("pDisk=%#p pIoCtx=%#p\n", pDisk, pIoCtx));
1254
1255 if (!ASMAtomicCmpXchgBool(&pDisk->fLocked, true, false))
1256 {
1257 Assert(pDisk->pIoCtxLockOwner != pIoCtx); /* No nesting allowed. */
1258
1259 rc = vdIoCtxDefer(pDisk, pIoCtx);
1260 if (RT_SUCCESS(rc))
1261 rc = VERR_VD_ASYNC_IO_IN_PROGRESS;
1262 }
1263 else
1264 {
1265 Assert(!pDisk->pIoCtxLockOwner);
1266 pDisk->pIoCtxLockOwner = pIoCtx;
1267 }
1268
1269 LogFlowFunc(("returns -> %Rrc\n", rc));
1270 return rc;
1271}
1272
1273static void vdIoCtxUnlockDisk(PVBOXHDD pDisk, PVDIOCTX pIoCtx, bool fProcessDeferredReqs)
1274{
1275 LogFlowFunc(("pDisk=%#p pIoCtx=%#p fProcessDeferredReqs=%RTbool\n",
1276 pDisk, pIoCtx, fProcessDeferredReqs));
1277
1278 LogFlow(("Unlocking disk lock owner is %#p\n", pDisk->pIoCtxLockOwner));
1279 Assert(pDisk->fLocked);
1280 Assert(pDisk->pIoCtxLockOwner == pIoCtx);
1281 pDisk->pIoCtxLockOwner = NULL;
1282 ASMAtomicXchgBool(&pDisk->fLocked, false);
1283
1284 if (fProcessDeferredReqs)
1285 {
1286 /* Process any pending writes if the current request didn't caused another growing. */
1287 RTCritSectEnter(&pDisk->CritSect);
1288
1289 if (!RTListIsEmpty(&pDisk->ListWriteLocked))
1290 {
1291 RTLISTNODE ListTmp;
1292
1293 RTListMove(&ListTmp, &pDisk->ListWriteLocked);
1294 RTCritSectLeave(&pDisk->CritSect);
1295
1296 /* Process the list. */
1297 do
1298 {
1299 int rc;
1300 PVDIOCTXDEFERRED pDeferred = RTListGetFirst(&ListTmp, VDIOCTXDEFERRED, NodeDeferred);
1301 PVDIOCTX pIoCtxWait = pDeferred->pIoCtx;
1302
1303 AssertPtr(pIoCtxWait);
1304
1305 RTListNodeRemove(&pDeferred->NodeDeferred);
1306 RTMemFree(pDeferred);
1307
1308 Assert(!pIoCtxWait->pIoCtxParent);
1309
1310 pIoCtxWait->fBlocked = false;
1311 LogFlowFunc(("Processing waiting I/O context pIoCtxWait=%#p\n", pIoCtxWait));
1312
1313 rc = vdIoCtxProcess(pIoCtxWait);
1314 if ( rc == VINF_VD_ASYNC_IO_FINISHED
1315 && ASMAtomicCmpXchgBool(&pIoCtxWait->fComplete, true, false))
1316 {
1317 LogFlowFunc(("Waiting I/O context completed pIoCtxWait=%#p\n", pIoCtxWait));
1318 vdThreadFinishWrite(pDisk);
1319 pIoCtxWait->Type.Root.pfnComplete(pIoCtxWait->Type.Root.pvUser1,
1320 pIoCtxWait->Type.Root.pvUser2,
1321 pIoCtxWait->rcReq);
1322 vdIoCtxFree(pDisk, pIoCtxWait);
1323 }
1324 } while (!RTListIsEmpty(&ListTmp));
1325 }
1326 else
1327 RTCritSectLeave(&pDisk->CritSect);
1328 }
1329
1330 LogFlowFunc(("returns\n"));
1331}
1332
1333/**
1334 * internal: read the specified amount of data in whatever blocks the backend
1335 * will give us - async version.
1336 */
1337static int vdReadHelperAsync(PVDIOCTX pIoCtx)
1338{
1339 int rc;
1340 size_t cbToRead = pIoCtx->cbTransfer;
1341 uint64_t uOffset = pIoCtx->uOffset;
1342 PVDIMAGE pCurrImage = NULL;
1343 size_t cbThisRead;
1344
1345 /* Loop until all reads started or we have a backend which needs to read metadata. */
1346 do
1347 {
1348 pCurrImage = pIoCtx->pImageCur;
1349
1350 /* Search for image with allocated block. Do not attempt to read more
1351 * than the previous reads marked as valid. Otherwise this would return
1352 * stale data when different block sizes are used for the images. */
1353 cbThisRead = cbToRead;
1354
1355 /*
1356 * Try to read from the given image.
1357 * If the block is not allocated read from override chain if present.
1358 */
1359 rc = pCurrImage->Backend->pfnAsyncRead(pCurrImage->pBackendData,
1360 uOffset, cbThisRead,
1361 pIoCtx, &cbThisRead);
1362
1363 if (rc == VERR_VD_BLOCK_FREE)
1364 {
1365 while ( pCurrImage->pPrev != NULL
1366 && rc == VERR_VD_BLOCK_FREE)
1367 {
1368 pCurrImage = pCurrImage->pPrev;
1369 rc = pCurrImage->Backend->pfnAsyncRead(pCurrImage->pBackendData,
1370 uOffset, cbThisRead,
1371 pIoCtx, &cbThisRead);
1372 }
1373 }
1374
1375 /* The task state will be updated on success already, don't do it here!. */
1376 if (rc == VERR_VD_BLOCK_FREE)
1377 {
1378 /* No image in the chain contains the data for the block. */
1379 vdIoCtxSet(pIoCtx, '\0', cbThisRead);
1380 ASMAtomicSubU32(&pIoCtx->cbTransferLeft, cbThisRead);
1381 rc = VINF_SUCCESS;
1382 }
1383 else if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
1384 rc = VINF_SUCCESS;
1385 else if (rc == VERR_VD_IOCTX_HALT)
1386 {
1387 uOffset += cbThisRead;
1388 cbToRead -= cbThisRead;
1389 pIoCtx->fBlocked = true;
1390 }
1391
1392 if (RT_FAILURE(rc))
1393 break;
1394
1395 cbToRead -= cbThisRead;
1396 uOffset += cbThisRead;
1397 } while (cbToRead != 0 && RT_SUCCESS(rc));
1398
1399 if ( rc == VERR_VD_NOT_ENOUGH_METADATA
1400 || rc == VERR_VD_IOCTX_HALT)
1401 {
1402 /* Save the current state. */
1403 pIoCtx->uOffset = uOffset;
1404 pIoCtx->cbTransfer = cbToRead;
1405 pIoCtx->pImageCur = pCurrImage ? pCurrImage : pIoCtx->pImageStart;
1406 }
1407
1408 return rc;
1409}
1410
1411/**
1412 * internal: parent image read wrapper for compacting.
1413 */
1414static int vdParentRead(void *pvUser, uint64_t uOffset, void *pvBuf,
1415 size_t cbRead)
1416{
1417 PVDPARENTSTATEDESC pParentState = (PVDPARENTSTATEDESC)pvUser;
1418 return vdReadHelper(pParentState->pDisk, pParentState->pImage, uOffset,
1419 pvBuf, cbRead, false /* fUpdateCache */);
1420}
1421
1422/**
1423 * internal: mark the disk as not modified.
1424 */
1425static void vdResetModifiedFlag(PVBOXHDD pDisk)
1426{
1427 if (pDisk->uModified & VD_IMAGE_MODIFIED_FLAG)
1428 {
1429 /* generate new last-modified uuid */
1430 if (!(pDisk->uModified & VD_IMAGE_MODIFIED_DISABLE_UUID_UPDATE))
1431 {
1432 RTUUID Uuid;
1433
1434 RTUuidCreate(&Uuid);
1435 pDisk->pLast->Backend->pfnSetModificationUuid(pDisk->pLast->pBackendData,
1436 &Uuid);
1437
1438 if (pDisk->pCache)
1439 pDisk->pCache->Backend->pfnSetModificationUuid(pDisk->pCache->pBackendData,
1440 &Uuid);
1441 }
1442
1443 pDisk->uModified &= ~VD_IMAGE_MODIFIED_FLAG;
1444 }
1445}
1446
1447/**
1448 * internal: mark the disk as modified.
1449 */
1450static void vdSetModifiedFlag(PVBOXHDD pDisk)
1451{
1452 pDisk->uModified |= VD_IMAGE_MODIFIED_FLAG;
1453 if (pDisk->uModified & VD_IMAGE_MODIFIED_FIRST)
1454 {
1455 pDisk->uModified &= ~VD_IMAGE_MODIFIED_FIRST;
1456
1457 /* First modify, so create a UUID and ensure it's written to disk. */
1458 vdResetModifiedFlag(pDisk);
1459
1460 if (!(pDisk->uModified & VD_IMAGE_MODIFIED_DISABLE_UUID_UPDATE))
1461 pDisk->pLast->Backend->pfnFlush(pDisk->pLast->pBackendData);
1462 }
1463}
1464
1465/**
1466 * internal: write a complete block (only used for diff images), taking the
1467 * remaining data from parent images. This implementation does not optimize
1468 * anything (except that it tries to read only that portions from parent
1469 * images that are really needed).
1470 */
1471static int vdWriteHelperStandard(PVBOXHDD pDisk, PVDIMAGE pImage,
1472 PVDIMAGE pImageParentOverride,
1473 uint64_t uOffset, size_t cbWrite,
1474 size_t cbThisWrite, size_t cbPreRead,
1475 size_t cbPostRead, const void *pvBuf,
1476 void *pvTmp)
1477{
1478 int rc = VINF_SUCCESS;
1479
1480 /* Read the data that goes before the write to fill the block. */
1481 if (cbPreRead)
1482 {
1483 /*
1484 * Updating the cache doesn't make sense here because
1485 * this will be done after the complete block was written.
1486 */
1487 rc = vdReadHelperEx(pDisk, pImage, pImageParentOverride,
1488 uOffset - cbPreRead, pvTmp, cbPreRead,
1489 true /* fZeroFreeBlocks*/,
1490 false /* fUpdateCache */, 0);
1491 if (RT_FAILURE(rc))
1492 return rc;
1493 }
1494
1495 /* Copy the data to the right place in the buffer. */
1496 memcpy((char *)pvTmp + cbPreRead, pvBuf, cbThisWrite);
1497
1498 /* Read the data that goes after the write to fill the block. */
1499 if (cbPostRead)
1500 {
1501 /* If we have data to be written, use that instead of reading
1502 * data from the image. */
1503 size_t cbWriteCopy;
1504 if (cbWrite > cbThisWrite)
1505 cbWriteCopy = RT_MIN(cbWrite - cbThisWrite, cbPostRead);
1506 else
1507 cbWriteCopy = 0;
1508 /* Figure out how much we cannot read from the image, because
1509 * the last block to write might exceed the nominal size of the
1510 * image for technical reasons. */
1511 size_t cbFill;
1512 if (uOffset + cbThisWrite + cbPostRead > pDisk->cbSize)
1513 cbFill = uOffset + cbThisWrite + cbPostRead - pDisk->cbSize;
1514 else
1515 cbFill = 0;
1516 /* The rest must be read from the image. */
1517 size_t cbReadImage = cbPostRead - cbWriteCopy - cbFill;
1518
1519 /* Now assemble the remaining data. */
1520 if (cbWriteCopy)
1521 memcpy((char *)pvTmp + cbPreRead + cbThisWrite,
1522 (char *)pvBuf + cbThisWrite, cbWriteCopy);
1523 if (cbReadImage)
1524 rc = vdReadHelperEx(pDisk, pImage, pImageParentOverride,
1525 uOffset + cbThisWrite + cbWriteCopy,
1526 (char *)pvTmp + cbPreRead + cbThisWrite + cbWriteCopy,
1527 cbReadImage, true /* fZeroFreeBlocks */,
1528 false /* fUpdateCache */, 0);
1529 if (RT_FAILURE(rc))
1530 return rc;
1531 /* Zero out the remainder of this block. Will never be visible, as this
1532 * is beyond the limit of the image. */
1533 if (cbFill)
1534 memset((char *)pvTmp + cbPreRead + cbThisWrite + cbWriteCopy + cbReadImage,
1535 '\0', cbFill);
1536 }
1537
1538 /* Write the full block to the virtual disk. */
1539 rc = pImage->Backend->pfnWrite(pImage->pBackendData,
1540 uOffset - cbPreRead, pvTmp,
1541 cbPreRead + cbThisWrite + cbPostRead,
1542 NULL, &cbPreRead, &cbPostRead, 0);
1543 Assert(rc != VERR_VD_BLOCK_FREE);
1544 Assert(cbPreRead == 0);
1545 Assert(cbPostRead == 0);
1546
1547 return rc;
1548}
1549
1550/**
1551 * internal: write a complete block (only used for diff images), taking the
1552 * remaining data from parent images. This implementation optimizes out writes
1553 * that do not change the data relative to the state as of the parent images.
1554 * All backends which support differential/growing images support this.
1555 */
1556static int vdWriteHelperOptimized(PVBOXHDD pDisk, PVDIMAGE pImage,
1557 PVDIMAGE pImageParentOverride,
1558 uint64_t uOffset, size_t cbWrite,
1559 size_t cbThisWrite, size_t cbPreRead,
1560 size_t cbPostRead, const void *pvBuf,
1561 void *pvTmp, unsigned cImagesRead)
1562{
1563 size_t cbFill = 0;
1564 size_t cbWriteCopy = 0;
1565 size_t cbReadImage = 0;
1566 int rc;
1567
1568 if (cbPostRead)
1569 {
1570 /* Figure out how much we cannot read from the image, because
1571 * the last block to write might exceed the nominal size of the
1572 * image for technical reasons. */
1573 if (uOffset + cbThisWrite + cbPostRead > pDisk->cbSize)
1574 cbFill = uOffset + cbThisWrite + cbPostRead - pDisk->cbSize;
1575
1576 /* If we have data to be written, use that instead of reading
1577 * data from the image. */
1578 if (cbWrite > cbThisWrite)
1579 cbWriteCopy = RT_MIN(cbWrite - cbThisWrite, cbPostRead);
1580
1581 /* The rest must be read from the image. */
1582 cbReadImage = cbPostRead - cbWriteCopy - cbFill;
1583 }
1584
1585 /* Read the entire data of the block so that we can compare whether it will
1586 * be modified by the write or not. */
1587 rc = vdReadHelperEx(pDisk, pImage, pImageParentOverride, uOffset - cbPreRead, pvTmp,
1588 cbPreRead + cbThisWrite + cbPostRead - cbFill,
1589 true /* fZeroFreeBlocks */, false /* fUpdateCache */,
1590 cImagesRead);
1591 if (RT_FAILURE(rc))
1592 return rc;
1593
1594 /* Check if the write would modify anything in this block. */
1595 if ( !memcmp((char *)pvTmp + cbPreRead, pvBuf, cbThisWrite)
1596 && (!cbWriteCopy || !memcmp((char *)pvTmp + cbPreRead + cbThisWrite,
1597 (char *)pvBuf + cbThisWrite, cbWriteCopy)))
1598 {
1599 /* Block is completely unchanged, so no need to write anything. */
1600 return VINF_SUCCESS;
1601 }
1602
1603 /* Copy the data to the right place in the buffer. */
1604 memcpy((char *)pvTmp + cbPreRead, pvBuf, cbThisWrite);
1605
1606 /* Handle the data that goes after the write to fill the block. */
1607 if (cbPostRead)
1608 {
1609 /* Now assemble the remaining data. */
1610 if (cbWriteCopy)
1611 memcpy((char *)pvTmp + cbPreRead + cbThisWrite,
1612 (char *)pvBuf + cbThisWrite, cbWriteCopy);
1613 /* Zero out the remainder of this block. Will never be visible, as this
1614 * is beyond the limit of the image. */
1615 if (cbFill)
1616 memset((char *)pvTmp + cbPreRead + cbThisWrite + cbWriteCopy + cbReadImage,
1617 '\0', cbFill);
1618 }
1619
1620 /* Write the full block to the virtual disk. */
1621 rc = pImage->Backend->pfnWrite(pImage->pBackendData,
1622 uOffset - cbPreRead, pvTmp,
1623 cbPreRead + cbThisWrite + cbPostRead,
1624 NULL, &cbPreRead, &cbPostRead, 0);
1625 Assert(rc != VERR_VD_BLOCK_FREE);
1626 Assert(cbPreRead == 0);
1627 Assert(cbPostRead == 0);
1628
1629 return rc;
1630}
1631
1632/**
1633 * internal: write buffer to the image, taking care of block boundaries and
1634 * write optimizations.
1635 */
1636static int vdWriteHelperEx(PVBOXHDD pDisk, PVDIMAGE pImage,
1637 PVDIMAGE pImageParentOverride, uint64_t uOffset,
1638 const void *pvBuf, size_t cbWrite,
1639 bool fUpdateCache, unsigned cImagesRead)
1640{
1641 int rc;
1642 unsigned fWrite;
1643 size_t cbThisWrite;
1644 size_t cbPreRead, cbPostRead;
1645 uint64_t uOffsetCur = uOffset;
1646 size_t cbWriteCur = cbWrite;
1647 const void *pcvBufCur = pvBuf;
1648
1649 /* Loop until all written. */
1650 do
1651 {
1652 /* Try to write the possibly partial block to the last opened image.
1653 * This works when the block is already allocated in this image or
1654 * if it is a full-block write (and allocation isn't suppressed below).
1655 * For image formats which don't support zero blocks, it's beneficial
1656 * to avoid unnecessarily allocating unchanged blocks. This prevents
1657 * unwanted expanding of images. VMDK is an example. */
1658 cbThisWrite = cbWriteCur;
1659 fWrite = (pImage->uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME)
1660 ? 0 : VD_WRITE_NO_ALLOC;
1661 rc = pImage->Backend->pfnWrite(pImage->pBackendData, uOffsetCur, pcvBufCur,
1662 cbThisWrite, &cbThisWrite, &cbPreRead,
1663 &cbPostRead, fWrite);
1664 if (rc == VERR_VD_BLOCK_FREE)
1665 {
1666 void *pvTmp = RTMemTmpAlloc(cbPreRead + cbThisWrite + cbPostRead);
1667 AssertBreakStmt(VALID_PTR(pvTmp), rc = VERR_NO_MEMORY);
1668
1669 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME))
1670 {
1671 /* Optimized write, suppress writing to a so far unallocated
1672 * block if the data is in fact not changed. */
1673 rc = vdWriteHelperOptimized(pDisk, pImage, pImageParentOverride,
1674 uOffsetCur, cbWriteCur,
1675 cbThisWrite, cbPreRead, cbPostRead,
1676 pcvBufCur, pvTmp, cImagesRead);
1677 }
1678 else
1679 {
1680 /* Normal write, not optimized in any way. The block will
1681 * be written no matter what. This will usually (unless the
1682 * backend has some further optimization enabled) cause the
1683 * block to be allocated. */
1684 rc = vdWriteHelperStandard(pDisk, pImage, pImageParentOverride,
1685 uOffsetCur, cbWriteCur,
1686 cbThisWrite, cbPreRead, cbPostRead,
1687 pcvBufCur, pvTmp);
1688 }
1689 RTMemTmpFree(pvTmp);
1690 if (RT_FAILURE(rc))
1691 break;
1692 }
1693
1694 cbWriteCur -= cbThisWrite;
1695 uOffsetCur += cbThisWrite;
1696 pcvBufCur = (char *)pcvBufCur + cbThisWrite;
1697 } while (cbWriteCur != 0 && RT_SUCCESS(rc));
1698
1699 /* Update the cache on success */
1700 if ( RT_SUCCESS(rc)
1701 && pDisk->pCache
1702 && fUpdateCache)
1703 rc = vdCacheWriteHelper(pDisk->pCache, uOffset, pvBuf, cbWrite, NULL);
1704
1705 return rc;
1706}
1707
1708/**
1709 * internal: write buffer to the image, taking care of block boundaries and
1710 * write optimizations.
1711 */
1712static int vdWriteHelper(PVBOXHDD pDisk, PVDIMAGE pImage, uint64_t uOffset,
1713 const void *pvBuf, size_t cbWrite, bool fUpdateCache)
1714{
1715 return vdWriteHelperEx(pDisk, pImage, NULL, uOffset, pvBuf, cbWrite,
1716 fUpdateCache, 0);
1717}
1718
1719/**
1720 * Internal: Copies the content of one disk to another one applying optimizations
1721 * to speed up the copy process if possible.
1722 */
1723static int vdCopyHelper(PVBOXHDD pDiskFrom, PVDIMAGE pImageFrom, PVBOXHDD pDiskTo,
1724 uint64_t cbSize, unsigned cImagesFromRead, unsigned cImagesToRead,
1725 bool fSuppressRedundantIo,
1726 PVDINTERFACE pIfProgress, PVDINTERFACEPROGRESS pCbProgress,
1727 PVDINTERFACE pDstIfProgress, PVDINTERFACEPROGRESS pDstCbProgress)
1728{
1729 int rc = VINF_SUCCESS;
1730 int rc2;
1731 uint64_t uOffset = 0;
1732 uint64_t cbRemaining = cbSize;
1733 void *pvBuf = NULL;
1734 bool fLockReadFrom = false;
1735 bool fLockWriteTo = false;
1736 bool fBlockwiseCopy = fSuppressRedundantIo || (cImagesFromRead > 0);
1737 unsigned uProgressOld = 0;
1738
1739 LogFlowFunc(("pDiskFrom=%#p pImageFrom=%#p pDiskTo=%#p cbSize=%llu cImagesFromRead=%u cImagesToRead=%u fSuppressRedundantIo=%RTbool pIfProgress=%#p pCbProgress=%#p pDstIfProgress=%#p pDstCbProgress=%#p\n",
1740 pDiskFrom, pImageFrom, pDiskTo, cbSize, cImagesFromRead, cImagesToRead, fSuppressRedundantIo, pIfProgress, pCbProgress, pDstIfProgress, pDstCbProgress));
1741
1742 /* Allocate tmp buffer. */
1743 pvBuf = RTMemTmpAlloc(VD_MERGE_BUFFER_SIZE);
1744 if (!pvBuf)
1745 return rc;
1746
1747 do
1748 {
1749 size_t cbThisRead = RT_MIN(VD_MERGE_BUFFER_SIZE, cbRemaining);
1750
1751 /* Note that we don't attempt to synchronize cross-disk accesses.
1752 * It wouldn't be very difficult to do, just the lock order would
1753 * need to be defined somehow to prevent deadlocks. Postpone such
1754 * magic as there is no use case for this. */
1755
1756 rc2 = vdThreadStartRead(pDiskFrom);
1757 AssertRC(rc2);
1758 fLockReadFrom = true;
1759
1760 if (fBlockwiseCopy)
1761 {
1762 /* Read the source data. */
1763 rc = pImageFrom->Backend->pfnRead(pImageFrom->pBackendData,
1764 uOffset, pvBuf, cbThisRead,
1765 &cbThisRead);
1766
1767 if ( rc == VERR_VD_BLOCK_FREE
1768 && cImagesFromRead != 1)
1769 {
1770 unsigned cImagesToProcess = cImagesFromRead;
1771
1772 for (PVDIMAGE pCurrImage = pImageFrom->pPrev;
1773 pCurrImage != NULL && rc == VERR_VD_BLOCK_FREE;
1774 pCurrImage = pCurrImage->pPrev)
1775 {
1776 rc = pCurrImage->Backend->pfnRead(pCurrImage->pBackendData,
1777 uOffset, pvBuf, cbThisRead,
1778 &cbThisRead);
1779 if (cImagesToProcess == 1)
1780 break;
1781 else if (cImagesToProcess > 0)
1782 cImagesToProcess--;
1783 }
1784 }
1785 }
1786 else
1787 rc = vdReadHelper(pDiskFrom, pImageFrom, uOffset, pvBuf, cbThisRead,
1788 false /* fUpdateCache */);
1789
1790 if (RT_FAILURE(rc) && rc != VERR_VD_BLOCK_FREE)
1791 break;
1792
1793 rc2 = vdThreadFinishRead(pDiskFrom);
1794 AssertRC(rc2);
1795 fLockReadFrom = false;
1796
1797 if (rc != VERR_VD_BLOCK_FREE)
1798 {
1799 rc2 = vdThreadStartWrite(pDiskTo);
1800 AssertRC(rc2);
1801 fLockWriteTo = true;
1802
1803 /* Only do collapsed I/O if we are copying the data blockwise. */
1804 rc = vdWriteHelperEx(pDiskTo, pDiskTo->pLast, NULL, uOffset, pvBuf,
1805 cbThisRead, false /* fUpdateCache */,
1806 fBlockwiseCopy ? cImagesToRead : 0);
1807 if (RT_FAILURE(rc))
1808 break;
1809
1810 rc2 = vdThreadFinishWrite(pDiskTo);
1811 AssertRC(rc2);
1812 fLockWriteTo = false;
1813 }
1814 else /* Don't propagate the error to the outside */
1815 rc = VINF_SUCCESS;
1816
1817 uOffset += cbThisRead;
1818 cbRemaining -= cbThisRead;
1819
1820 unsigned uProgressNew = uOffset * 99 / cbSize;
1821 if (uProgressNew != uProgressOld)
1822 {
1823 uProgressOld = uProgressNew;
1824
1825 if (pCbProgress && pCbProgress->pfnProgress)
1826 {
1827 rc = pCbProgress->pfnProgress(pIfProgress->pvUser,
1828 uProgressOld);
1829 if (RT_FAILURE(rc))
1830 break;
1831 }
1832 if (pDstCbProgress && pDstCbProgress->pfnProgress)
1833 {
1834 rc = pDstCbProgress->pfnProgress(pDstIfProgress->pvUser,
1835 uProgressOld);
1836 if (RT_FAILURE(rc))
1837 break;
1838 }
1839 }
1840 } while (uOffset < cbSize);
1841
1842 RTMemFree(pvBuf);
1843
1844 if (fLockReadFrom)
1845 {
1846 rc2 = vdThreadFinishRead(pDiskFrom);
1847 AssertRC(rc2);
1848 }
1849
1850 if (fLockWriteTo)
1851 {
1852 rc2 = vdThreadFinishWrite(pDiskTo);
1853 AssertRC(rc2);
1854 }
1855
1856 LogFlowFunc(("returns rc=%Rrc\n", rc));
1857 return rc;
1858}
1859
1860/**
1861 * Flush helper async version.
1862 */
1863static int vdSetModifiedHelperAsync(PVDIOCTX pIoCtx)
1864{
1865 int rc = VINF_SUCCESS;
1866 PVBOXHDD pDisk = pIoCtx->pDisk;
1867 PVDIMAGE pImage = pIoCtx->pImageCur;
1868
1869 rc = pImage->Backend->pfnAsyncFlush(pImage->pBackendData, pIoCtx);
1870 if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
1871 rc = VINF_SUCCESS;
1872
1873 return rc;
1874}
1875
1876/**
1877 * internal: mark the disk as modified - async version.
1878 */
1879static int vdSetModifiedFlagAsync(PVBOXHDD pDisk, PVDIOCTX pIoCtx)
1880{
1881 int rc = VINF_SUCCESS;
1882
1883 pDisk->uModified |= VD_IMAGE_MODIFIED_FLAG;
1884 if (pDisk->uModified & VD_IMAGE_MODIFIED_FIRST)
1885 {
1886 rc = vdIoCtxLockDisk(pDisk, pIoCtx);
1887 if (RT_SUCCESS(rc))
1888 {
1889 pDisk->uModified &= ~VD_IMAGE_MODIFIED_FIRST;
1890
1891 /* First modify, so create a UUID and ensure it's written to disk. */
1892 vdResetModifiedFlag(pDisk);
1893
1894 if (!(pDisk->uModified & VD_IMAGE_MODIFIED_DISABLE_UUID_UPDATE))
1895 {
1896 PVDIOCTX pIoCtxFlush = vdIoCtxChildAlloc(pDisk, VDIOCTXTXDIR_FLUSH,
1897 0, 0, pDisk->pLast,
1898 NULL, pIoCtx, 0, 0, NULL,
1899 vdSetModifiedHelperAsync);
1900
1901 if (pIoCtxFlush)
1902 {
1903 rc = vdIoCtxProcess(pIoCtxFlush);
1904 if (rc == VINF_VD_ASYNC_IO_FINISHED)
1905 {
1906 vdIoCtxUnlockDisk(pDisk, pIoCtx, false /* fProcessDeferredReqs */);
1907 vdIoCtxFree(pDisk, pIoCtxFlush);
1908 }
1909 else if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
1910 {
1911 pIoCtx->fBlocked = true;
1912 }
1913 else /* Another error */
1914 vdIoCtxFree(pDisk, pIoCtxFlush);
1915 }
1916 else
1917 rc = VERR_NO_MEMORY;
1918 }
1919 }
1920 }
1921
1922 return rc;
1923}
1924
1925/**
1926 * internal: write a complete block (only used for diff images), taking the
1927 * remaining data from parent images. This implementation does not optimize
1928 * anything (except that it tries to read only that portions from parent
1929 * images that are really needed) - async version.
1930 */
1931static int vdWriteHelperStandardAsync(PVDIOCTX pIoCtx)
1932{
1933 int rc = VINF_SUCCESS;
1934
1935#if 0
1936
1937 /* Read the data that goes before the write to fill the block. */
1938 if (cbPreRead)
1939 {
1940 rc = vdReadHelperAsync(pIoCtxDst);
1941 if (RT_FAILURE(rc))
1942 return rc;
1943 }
1944
1945 /* Copy the data to the right place in the buffer. */
1946 vdIoCtxCopy(pIoCtxDst, pIoCtxSrc, cbThisWrite);
1947
1948 /* Read the data that goes after the write to fill the block. */
1949 if (cbPostRead)
1950 {
1951 /* If we have data to be written, use that instead of reading
1952 * data from the image. */
1953 size_t cbWriteCopy;
1954 if (cbWrite > cbThisWrite)
1955 cbWriteCopy = RT_MIN(cbWrite - cbThisWrite, cbPostRead);
1956 else
1957 cbWriteCopy = 0;
1958 /* Figure out how much we cannot read from the image, because
1959 * the last block to write might exceed the nominal size of the
1960 * image for technical reasons. */
1961 size_t cbFill;
1962 if (uOffset + cbThisWrite + cbPostRead > pDisk->cbSize)
1963 cbFill = uOffset + cbThisWrite + cbPostRead - pDisk->cbSize;
1964 else
1965 cbFill = 0;
1966 /* The rest must be read from the image. */
1967 size_t cbReadImage = cbPostRead - cbWriteCopy - cbFill;
1968
1969 /* Now assemble the remaining data. */
1970 if (cbWriteCopy)
1971 {
1972 vdIoCtxCopy(pIoCtxDst, pIoCtxSrc, cbWriteCopy);
1973 ASMAtomicSubU32(&pIoCtxDst->cbTransferLeft, cbWriteCopy);
1974 }
1975
1976 if (cbReadImage)
1977 rc = vdReadHelperAsync(pDisk, pImage, pImageParentOverride, pIoCtxDst,
1978 uOffset + cbThisWrite + cbWriteCopy,
1979 cbReadImage);
1980 if (RT_FAILURE(rc))
1981 return rc;
1982 /* Zero out the remainder of this block. Will never be visible, as this
1983 * is beyond the limit of the image. */
1984 if (cbFill)
1985 {
1986 vdIoCtxSet(pIoCtxDst, '\0', cbFill);
1987 ASMAtomicSubU32(&pIoCtxDst->cbTransferLeft, cbFill);
1988 }
1989 }
1990
1991 if ( !pIoCtxDst->cbTransferLeft
1992 && !pIoCtxDst->cMetaTransfersPending
1993 && ASMAtomicCmpXchgBool(&pIoCtxDst->fComplete, true, false))
1994 {
1995 /* Write the full block to the virtual disk. */
1996 vdIoCtxChildReset(pIoCtxDst);
1997 rc = pImage->Backend->pfnAsyncWrite(pImage->pBackendData,
1998 uOffset - cbPreRead,
1999 cbPreRead + cbThisWrite + cbPostRead,
2000 pIoCtxDst,
2001 NULL, &cbPreRead, &cbPostRead, 0);
2002 Assert(rc != VERR_VD_BLOCK_FREE);
2003 Assert(cbPreRead == 0);
2004 Assert(cbPostRead == 0);
2005 }
2006 else
2007 {
2008 LogFlow(("cbTransferLeft=%u cMetaTransfersPending=%u fComplete=%RTbool\n",
2009 pIoCtxDst->cbTransferLeft, pIoCtxDst->cMetaTransfersPending,
2010 pIoCtxDst->fComplete));
2011 rc = VERR_VD_ASYNC_IO_IN_PROGRESS;
2012 }
2013
2014 return rc;
2015#endif
2016 return VERR_NOT_IMPLEMENTED;
2017}
2018
2019static int vdWriteHelperOptimizedCommitAsync(PVDIOCTX pIoCtx)
2020{
2021 int rc = VINF_SUCCESS;
2022 PVDIMAGE pImage = pIoCtx->pImageStart;
2023 size_t cbPreRead = pIoCtx->Type.Child.cbPreRead;
2024 size_t cbPostRead = pIoCtx->Type.Child.cbPostRead;
2025 size_t cbThisWrite = pIoCtx->Type.Child.cbTransferParent;
2026
2027 LogFlowFunc(("pIoCtx=%#p\n", pIoCtx));
2028 rc = pImage->Backend->pfnAsyncWrite(pImage->pBackendData,
2029 pIoCtx->uOffset - cbPreRead,
2030 cbPreRead + cbThisWrite + cbPostRead,
2031 pIoCtx, NULL, &cbPreRead, &cbPostRead, 0);
2032 Assert(rc != VERR_VD_BLOCK_FREE);
2033 Assert(rc == VERR_VD_NOT_ENOUGH_METADATA || cbPreRead == 0);
2034 Assert(rc == VERR_VD_NOT_ENOUGH_METADATA || cbPostRead == 0);
2035 if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
2036 rc = VINF_SUCCESS;
2037 else if (rc == VERR_VD_IOCTX_HALT)
2038 {
2039 pIoCtx->fBlocked = true;
2040 rc = VINF_SUCCESS;
2041 }
2042
2043 LogFlowFunc(("returns rc=%Rrc\n", rc));
2044 return rc;
2045}
2046
2047static int vdWriteHelperOptimizedCmpAndWriteAsync(PVDIOCTX pIoCtx)
2048{
2049 int rc = VINF_SUCCESS;
2050 PVDIMAGE pImage = pIoCtx->pImageCur;
2051 size_t cbThisWrite = 0;
2052 size_t cbPreRead = pIoCtx->Type.Child.cbPreRead;
2053 size_t cbPostRead = pIoCtx->Type.Child.cbPostRead;
2054 size_t cbWriteCopy = pIoCtx->Type.Child.Write.Optimized.cbWriteCopy;
2055 size_t cbFill = pIoCtx->Type.Child.Write.Optimized.cbFill;
2056 size_t cbReadImage = pIoCtx->Type.Child.Write.Optimized.cbReadImage;
2057 PVDIOCTX pIoCtxParent = pIoCtx->pIoCtxParent;
2058
2059 LogFlowFunc(("pIoCtx=%#p\n", pIoCtx));
2060
2061 AssertPtr(pIoCtxParent);
2062 Assert(!pIoCtxParent->pIoCtxParent);
2063 Assert(!pIoCtx->cbTransferLeft && !pIoCtx->cMetaTransfersPending);
2064
2065 vdIoCtxChildReset(pIoCtx);
2066 cbThisWrite = pIoCtx->Type.Child.cbTransferParent;
2067 RTSgBufAdvance(&pIoCtx->SgBuf, cbPreRead);
2068
2069 /* Check if the write would modify anything in this block. */
2070 if (!RTSgBufCmp(&pIoCtx->SgBuf, &pIoCtxParent->SgBuf, cbThisWrite))
2071 {
2072 RTSGBUF SgBufSrcTmp;
2073
2074 RTSgBufClone(&SgBufSrcTmp, &pIoCtxParent->SgBuf);
2075 RTSgBufAdvance(&SgBufSrcTmp, cbThisWrite);
2076 RTSgBufAdvance(&pIoCtx->SgBuf, cbThisWrite);
2077
2078 if (!cbWriteCopy || !RTSgBufCmp(&pIoCtx->SgBuf, &SgBufSrcTmp, cbWriteCopy))
2079 {
2080 /* Block is completely unchanged, so no need to write anything. */
2081 LogFlowFunc(("Block didn't changed\n"));
2082 ASMAtomicWriteU32(&pIoCtx->cbTransferLeft, 0);
2083 RTSgBufAdvance(&pIoCtxParent->SgBuf, cbThisWrite);
2084 return VINF_VD_ASYNC_IO_FINISHED;
2085 }
2086 }
2087
2088 /* Copy the data to the right place in the buffer. */
2089 RTSgBufReset(&pIoCtx->SgBuf);
2090 RTSgBufAdvance(&pIoCtx->SgBuf, cbPreRead);
2091 vdIoCtxCopy(pIoCtx, pIoCtxParent, cbThisWrite);
2092
2093 /* Handle the data that goes after the write to fill the block. */
2094 if (cbPostRead)
2095 {
2096 /* Now assemble the remaining data. */
2097 if (cbWriteCopy)
2098 {
2099 /*
2100 * The S/G buffer of the parent needs to be cloned because
2101 * it is not allowed to modify the state.
2102 */
2103 RTSGBUF SgBufParentTmp;
2104
2105 RTSgBufClone(&SgBufParentTmp, &pIoCtxParent->SgBuf);
2106 RTSgBufCopy(&pIoCtx->SgBuf, &SgBufParentTmp, cbWriteCopy);
2107 }
2108
2109 /* Zero out the remainder of this block. Will never be visible, as this
2110 * is beyond the limit of the image. */
2111 if (cbFill)
2112 {
2113 RTSgBufAdvance(&pIoCtx->SgBuf, cbReadImage);
2114 vdIoCtxSet(pIoCtx, '\0', cbFill);
2115 }
2116 }
2117
2118 /* Write the full block to the virtual disk. */
2119 RTSgBufReset(&pIoCtx->SgBuf);
2120 pIoCtx->pfnIoCtxTransferNext = vdWriteHelperOptimizedCommitAsync;
2121
2122 return rc;
2123}
2124
2125static int vdWriteHelperOptimizedPreReadAsync(PVDIOCTX pIoCtx)
2126{
2127 int rc = VINF_SUCCESS;
2128
2129 LogFlowFunc(("pIoCtx=%#p\n", pIoCtx));
2130
2131 if (pIoCtx->cbTransferLeft)
2132 rc = vdReadHelperAsync(pIoCtx);
2133
2134 if ( RT_SUCCESS(rc)
2135 && ( pIoCtx->cbTransferLeft
2136 || pIoCtx->cMetaTransfersPending))
2137 rc = VERR_VD_ASYNC_IO_IN_PROGRESS;
2138 else
2139 pIoCtx->pfnIoCtxTransferNext = vdWriteHelperOptimizedCmpAndWriteAsync;
2140
2141 return rc;
2142}
2143
2144/**
2145 * internal: write a complete block (only used for diff images), taking the
2146 * remaining data from parent images. This implementation optimizes out writes
2147 * that do not change the data relative to the state as of the parent images.
2148 * All backends which support differential/growing images support this - async version.
2149 */
2150static int vdWriteHelperOptimizedAsync(PVDIOCTX pIoCtx)
2151{
2152 PVBOXHDD pDisk = pIoCtx->pDisk;
2153 uint64_t uOffset = pIoCtx->Type.Child.uOffsetSaved;
2154 size_t cbThisWrite = pIoCtx->Type.Child.cbTransferParent;
2155 size_t cbPreRead = pIoCtx->Type.Child.cbPreRead;
2156 size_t cbPostRead = pIoCtx->Type.Child.cbPostRead;
2157 size_t cbWrite = pIoCtx->Type.Child.cbWriteParent;
2158 size_t cbFill = 0;
2159 size_t cbWriteCopy = 0;
2160 size_t cbReadImage = 0;
2161
2162 LogFlowFunc(("pIoCtx=%#p\n", pIoCtx));
2163
2164 AssertPtr(pIoCtx->pIoCtxParent);
2165 Assert(!pIoCtx->pIoCtxParent->pIoCtxParent);
2166
2167 if (cbPostRead)
2168 {
2169 /* Figure out how much we cannot read from the image, because
2170 * the last block to write might exceed the nominal size of the
2171 * image for technical reasons. */
2172 if (uOffset + cbThisWrite + cbPostRead > pDisk->cbSize)
2173 cbFill = uOffset + cbThisWrite + cbPostRead - pDisk->cbSize;
2174
2175 /* If we have data to be written, use that instead of reading
2176 * data from the image. */
2177 if (cbWrite > cbThisWrite)
2178 cbWriteCopy = RT_MIN(cbWrite - cbThisWrite, cbPostRead);
2179
2180 /* The rest must be read from the image. */
2181 cbReadImage = cbPostRead - cbWriteCopy - cbFill;
2182 }
2183
2184 pIoCtx->Type.Child.Write.Optimized.cbFill = cbFill;
2185 pIoCtx->Type.Child.Write.Optimized.cbWriteCopy = cbWriteCopy;
2186 pIoCtx->Type.Child.Write.Optimized.cbReadImage = cbReadImage;
2187
2188 /* Read the entire data of the block so that we can compare whether it will
2189 * be modified by the write or not. */
2190 pIoCtx->cbTransferLeft = cbPreRead + cbThisWrite + cbPostRead - cbFill;
2191 pIoCtx->cbTransfer = pIoCtx->cbTransferLeft;
2192 pIoCtx->uOffset -= cbPreRead;
2193
2194 /* Next step */
2195 pIoCtx->pfnIoCtxTransferNext = vdWriteHelperOptimizedPreReadAsync;
2196 return VINF_SUCCESS;
2197}
2198
2199/**
2200 * internal: write buffer to the image, taking care of block boundaries and
2201 * write optimizations - async version.
2202 */
2203static int vdWriteHelperAsync(PVDIOCTX pIoCtx)
2204{
2205 int rc;
2206 size_t cbWrite = pIoCtx->cbTransfer;
2207 uint64_t uOffset = pIoCtx->uOffset;
2208 PVDIMAGE pImage = pIoCtx->pImageCur;
2209 PVBOXHDD pDisk = pIoCtx->pDisk;
2210 unsigned fWrite;
2211 size_t cbThisWrite;
2212 size_t cbPreRead, cbPostRead;
2213
2214 rc = vdSetModifiedFlagAsync(pDisk, pIoCtx);
2215 if (RT_FAILURE(rc)) /* Includes I/O in progress. */
2216 return rc;
2217
2218 /* Loop until all written. */
2219 do
2220 {
2221 /* Try to write the possibly partial block to the last opened image.
2222 * This works when the block is already allocated in this image or
2223 * if it is a full-block write (and allocation isn't suppressed below).
2224 * For image formats which don't support zero blocks, it's beneficial
2225 * to avoid unnecessarily allocating unchanged blocks. This prevents
2226 * unwanted expanding of images. VMDK is an example. */
2227 cbThisWrite = cbWrite;
2228 fWrite = (pImage->uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME)
2229 ? 0 : VD_WRITE_NO_ALLOC;
2230 rc = pImage->Backend->pfnAsyncWrite(pImage->pBackendData, uOffset,
2231 cbThisWrite, pIoCtx,
2232 &cbThisWrite, &cbPreRead,
2233 &cbPostRead, fWrite);
2234 if (rc == VERR_VD_BLOCK_FREE)
2235 {
2236 /* Lock the disk .*/
2237 rc = vdIoCtxLockDisk(pDisk, pIoCtx);
2238 if (RT_SUCCESS(rc))
2239 {
2240 /*
2241 * Allocate segment and buffer in one go.
2242 * A bit hackish but avoids the need to allocate memory twice.
2243 */
2244 PRTSGBUF pTmp = (PRTSGBUF)RTMemAlloc(cbPreRead + cbThisWrite + cbPostRead + sizeof(RTSGSEG) + sizeof(RTSGBUF));
2245 AssertBreakStmt(VALID_PTR(pTmp), rc = VERR_NO_MEMORY);
2246 PRTSGSEG pSeg = (PRTSGSEG)(pTmp + 1);
2247
2248 pSeg->pvSeg = pSeg + 1;
2249 pSeg->cbSeg = cbPreRead + cbThisWrite + cbPostRead;
2250 RTSgBufInit(pTmp, pSeg, 1);
2251
2252 PVDIOCTX pIoCtxWrite = vdIoCtxChildAlloc(pDisk, VDIOCTXTXDIR_WRITE,
2253 uOffset, pSeg->cbSeg, pImage,
2254 pTmp,
2255 pIoCtx, cbThisWrite,
2256 cbWrite,
2257 pTmp,
2258 (pImage->uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME)
2259 ? vdWriteHelperStandardAsync
2260 : vdWriteHelperOptimizedAsync);
2261 if (!VALID_PTR(pIoCtxWrite))
2262 {
2263 RTMemTmpFree(pTmp);
2264 rc = VERR_NO_MEMORY;
2265 break;
2266 }
2267
2268 LogFlowFunc(("Disk is growing because of pIoCtx=%#p pIoCtxWrite=%#p\n",
2269 pIoCtx, pIoCtxWrite));
2270
2271 pIoCtxWrite->Type.Child.cbPreRead = cbPreRead;
2272 pIoCtxWrite->Type.Child.cbPostRead = cbPostRead;
2273
2274 /* Process the write request */
2275 rc = vdIoCtxProcess(pIoCtxWrite);
2276
2277 if (RT_FAILURE(rc) && (rc != VERR_VD_ASYNC_IO_IN_PROGRESS))
2278 {
2279 vdIoCtxFree(pDisk, pIoCtxWrite);
2280 break;
2281 }
2282 else if ( rc == VINF_VD_ASYNC_IO_FINISHED
2283 && ASMAtomicCmpXchgBool(&pIoCtxWrite->fComplete, true, false))
2284 {
2285 LogFlow(("Child write request completed\n"));
2286 Assert(pIoCtx->cbTransferLeft >= cbThisWrite);
2287 ASMAtomicSubU32(&pIoCtx->cbTransferLeft, cbThisWrite);
2288 vdIoCtxUnlockDisk(pDisk, pIoCtx, false /* fProcessDeferredReqs*/ );
2289 vdIoCtxFree(pDisk, pIoCtxWrite);
2290
2291 rc = VINF_SUCCESS;
2292 }
2293 else
2294 {
2295 LogFlow(("Child write pending\n"));
2296 pIoCtx->fBlocked = true;
2297 rc = VERR_VD_ASYNC_IO_IN_PROGRESS;
2298 cbWrite -= cbThisWrite;
2299 uOffset += cbThisWrite;
2300 break;
2301 }
2302 }
2303 else
2304 {
2305 rc = VERR_VD_ASYNC_IO_IN_PROGRESS;
2306 break;
2307 }
2308 }
2309
2310 if (rc == VERR_VD_IOCTX_HALT)
2311 {
2312 cbWrite -= cbThisWrite;
2313 uOffset += cbThisWrite;
2314 pIoCtx->fBlocked = true;
2315 break;
2316 }
2317 else if (rc == VERR_VD_NOT_ENOUGH_METADATA)
2318 break;
2319
2320 cbWrite -= cbThisWrite;
2321 uOffset += cbThisWrite;
2322 } while (cbWrite != 0 && (RT_SUCCESS(rc) || rc == VERR_VD_ASYNC_IO_IN_PROGRESS));
2323
2324 if ( rc == VERR_VD_ASYNC_IO_IN_PROGRESS
2325 || rc == VERR_VD_NOT_ENOUGH_METADATA
2326 || rc == VERR_VD_IOCTX_HALT)
2327 {
2328 /*
2329 * Tell the caller that we don't need to go back here because all
2330 * writes are initiated.
2331 */
2332 if (!cbWrite)
2333 rc = VINF_SUCCESS;
2334
2335 pIoCtx->uOffset = uOffset;
2336 pIoCtx->cbTransfer = cbWrite;
2337 }
2338
2339 return rc;
2340}
2341
2342/**
2343 * Flush helper async version.
2344 */
2345static int vdFlushHelperAsync(PVDIOCTX pIoCtx)
2346{
2347 int rc = VINF_SUCCESS;
2348 PVBOXHDD pDisk = pIoCtx->pDisk;
2349 PVDIMAGE pImage = pIoCtx->pImageCur;
2350
2351 rc = vdIoCtxLockDisk(pDisk, pIoCtx);
2352 if (RT_SUCCESS(rc))
2353 {
2354 vdResetModifiedFlag(pDisk);
2355 rc = pImage->Backend->pfnAsyncFlush(pImage->pBackendData, pIoCtx);
2356 if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
2357 rc = VINF_SUCCESS;
2358 else if (rc == VINF_VD_ASYNC_IO_FINISHED)
2359 vdIoCtxUnlockDisk(pDisk, pIoCtx, true /* fProcessDeferredReqs */);
2360 }
2361
2362 return rc;
2363}
2364
2365/**
2366 * internal: scans plugin directory and loads the backends have been found.
2367 */
2368static int vdLoadDynamicBackends()
2369{
2370#ifndef VBOX_HDD_NO_DYNAMIC_BACKENDS
2371 int rc = VINF_SUCCESS;
2372 PRTDIR pPluginDir = NULL;
2373
2374 /* Enumerate plugin backends. */
2375 char szPath[RTPATH_MAX];
2376 rc = RTPathAppPrivateArch(szPath, sizeof(szPath));
2377 if (RT_FAILURE(rc))
2378 return rc;
2379
2380 /* To get all entries with VBoxHDD as prefix. */
2381 char *pszPluginFilter = RTPathJoinA(szPath, VBOX_HDDFORMAT_PLUGIN_PREFIX "*");
2382 if (!pszPluginFilter)
2383 return VERR_NO_STR_MEMORY;
2384
2385 PRTDIRENTRYEX pPluginDirEntry = NULL;
2386 size_t cbPluginDirEntry = sizeof(RTDIRENTRYEX);
2387 /* The plugins are in the same directory as the other shared libs. */
2388 rc = RTDirOpenFiltered(&pPluginDir, pszPluginFilter, RTDIRFILTER_WINNT);
2389 if (RT_FAILURE(rc))
2390 {
2391 /* On Windows the above immediately signals that there are no
2392 * files matching, while on other platforms enumerating the
2393 * files below fails. Either way: no plugins. */
2394 goto out;
2395 }
2396
2397 pPluginDirEntry = (PRTDIRENTRYEX)RTMemAllocZ(sizeof(RTDIRENTRYEX));
2398 if (!pPluginDirEntry)
2399 {
2400 rc = VERR_NO_MEMORY;
2401 goto out;
2402 }
2403
2404 while ((rc = RTDirReadEx(pPluginDir, pPluginDirEntry, &cbPluginDirEntry, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK)) != VERR_NO_MORE_FILES)
2405 {
2406 RTLDRMOD hPlugin = NIL_RTLDRMOD;
2407 PFNVBOXHDDFORMATLOAD pfnHDDFormatLoad = NULL;
2408 PVBOXHDDBACKEND pBackend = NULL;
2409 char *pszPluginPath = NULL;
2410
2411 if (rc == VERR_BUFFER_OVERFLOW)
2412 {
2413 /* allocate new buffer. */
2414 RTMemFree(pPluginDirEntry);
2415 pPluginDirEntry = (PRTDIRENTRYEX)RTMemAllocZ(cbPluginDirEntry);
2416 if (!pPluginDirEntry)
2417 {
2418 rc = VERR_NO_MEMORY;
2419 break;
2420 }
2421 /* Retry. */
2422 rc = RTDirReadEx(pPluginDir, pPluginDirEntry, &cbPluginDirEntry, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK);
2423 if (RT_FAILURE(rc))
2424 break;
2425 }
2426 else if (RT_FAILURE(rc))
2427 break;
2428
2429 /* We got the new entry. */
2430 if (!RTFS_IS_FILE(pPluginDirEntry->Info.Attr.fMode))
2431 continue;
2432
2433 /* Prepend the path to the libraries. */
2434 pszPluginPath = RTPathJoinA(szPath, pPluginDirEntry->szName);
2435 if (!pszPluginPath)
2436 {
2437 rc = VERR_NO_STR_MEMORY;
2438 break;
2439 }
2440
2441 rc = SUPR3HardenedLdrLoadPlugIn(pszPluginPath, &hPlugin, NULL);
2442 if (RT_SUCCESS(rc))
2443 {
2444 rc = RTLdrGetSymbol(hPlugin, VBOX_HDDFORMAT_LOAD_NAME, (void**)&pfnHDDFormatLoad);
2445 if (RT_FAILURE(rc) || !pfnHDDFormatLoad)
2446 {
2447 LogFunc(("error resolving the entry point %s in plugin %s, rc=%Rrc, pfnHDDFormat=%#p\n", VBOX_HDDFORMAT_LOAD_NAME, pPluginDirEntry->szName, rc, pfnHDDFormatLoad));
2448 if (RT_SUCCESS(rc))
2449 rc = VERR_SYMBOL_NOT_FOUND;
2450 }
2451
2452 if (RT_SUCCESS(rc))
2453 {
2454 /* Get the function table. */
2455 rc = pfnHDDFormatLoad(&pBackend);
2456 if (RT_SUCCESS(rc) && pBackend->cbSize == sizeof(VBOXHDDBACKEND))
2457 {
2458 pBackend->hPlugin = hPlugin;
2459 vdAddBackend(pBackend);
2460 }
2461 else
2462 LogFunc(("ignored plugin '%s': pBackend->cbSize=%d rc=%Rrc\n", pszPluginPath, pBackend->cbSize, rc));
2463 }
2464 else
2465 LogFunc(("ignored plugin '%s': rc=%Rrc\n", pszPluginPath, rc));
2466
2467 if (RT_FAILURE(rc))
2468 RTLdrClose(hPlugin);
2469 }
2470 RTStrFree(pszPluginPath);
2471 }
2472out:
2473 if (rc == VERR_NO_MORE_FILES)
2474 rc = VINF_SUCCESS;
2475 RTStrFree(pszPluginFilter);
2476 if (pPluginDirEntry)
2477 RTMemFree(pPluginDirEntry);
2478 if (pPluginDir)
2479 RTDirClose(pPluginDir);
2480 return rc;
2481#else
2482 return VINF_SUCCESS;
2483#endif
2484}
2485
2486/**
2487 * internal: scans plugin directory and loads the cache backends have been found.
2488 */
2489static int vdLoadDynamicCacheBackends()
2490{
2491#ifndef VBOX_HDD_NO_DYNAMIC_BACKENDS
2492 int rc = VINF_SUCCESS;
2493 PRTDIR pPluginDir = NULL;
2494
2495 /* Enumerate plugin backends. */
2496 char szPath[RTPATH_MAX];
2497 rc = RTPathAppPrivateArch(szPath, sizeof(szPath));
2498 if (RT_FAILURE(rc))
2499 return rc;
2500
2501 /* To get all entries with VBoxHDD as prefix. */
2502 char *pszPluginFilter = RTPathJoinA(szPath, VD_CACHEFORMAT_PLUGIN_PREFIX "*");
2503 if (!pszPluginFilter)
2504 {
2505 rc = VERR_NO_STR_MEMORY;
2506 return rc;
2507 }
2508
2509 PRTDIRENTRYEX pPluginDirEntry = NULL;
2510 size_t cbPluginDirEntry = sizeof(RTDIRENTRYEX);
2511 /* The plugins are in the same directory as the other shared libs. */
2512 rc = RTDirOpenFiltered(&pPluginDir, pszPluginFilter, RTDIRFILTER_WINNT);
2513 if (RT_FAILURE(rc))
2514 {
2515 /* On Windows the above immediately signals that there are no
2516 * files matching, while on other platforms enumerating the
2517 * files below fails. Either way: no plugins. */
2518 goto out;
2519 }
2520
2521 pPluginDirEntry = (PRTDIRENTRYEX)RTMemAllocZ(sizeof(RTDIRENTRYEX));
2522 if (!pPluginDirEntry)
2523 {
2524 rc = VERR_NO_MEMORY;
2525 goto out;
2526 }
2527
2528 while ((rc = RTDirReadEx(pPluginDir, pPluginDirEntry, &cbPluginDirEntry, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK)) != VERR_NO_MORE_FILES)
2529 {
2530 RTLDRMOD hPlugin = NIL_RTLDRMOD;
2531 PFNVDCACHEFORMATLOAD pfnVDCacheLoad = NULL;
2532 PVDCACHEBACKEND pBackend = NULL;
2533 char *pszPluginPath = NULL;
2534
2535 if (rc == VERR_BUFFER_OVERFLOW)
2536 {
2537 /* allocate new buffer. */
2538 RTMemFree(pPluginDirEntry);
2539 pPluginDirEntry = (PRTDIRENTRYEX)RTMemAllocZ(cbPluginDirEntry);
2540 if (!pPluginDirEntry)
2541 {
2542 rc = VERR_NO_MEMORY;
2543 break;
2544 }
2545 /* Retry. */
2546 rc = RTDirReadEx(pPluginDir, pPluginDirEntry, &cbPluginDirEntry, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK);
2547 if (RT_FAILURE(rc))
2548 break;
2549 }
2550 else if (RT_FAILURE(rc))
2551 break;
2552
2553 /* We got the new entry. */
2554 if (!RTFS_IS_FILE(pPluginDirEntry->Info.Attr.fMode))
2555 continue;
2556
2557 /* Prepend the path to the libraries. */
2558 pszPluginPath = RTPathJoinA(szPath, pPluginDirEntry->szName);
2559 if (!pszPluginPath)
2560 {
2561 rc = VERR_NO_STR_MEMORY;
2562 break;
2563 }
2564
2565 rc = SUPR3HardenedLdrLoadPlugIn(pszPluginPath, &hPlugin, NULL);
2566 if (RT_SUCCESS(rc))
2567 {
2568 rc = RTLdrGetSymbol(hPlugin, VD_CACHEFORMAT_LOAD_NAME, (void**)&pfnVDCacheLoad);
2569 if (RT_FAILURE(rc) || !pfnVDCacheLoad)
2570 {
2571 LogFunc(("error resolving the entry point %s in plugin %s, rc=%Rrc, pfnVDCacheLoad=%#p\n",
2572 VD_CACHEFORMAT_LOAD_NAME, pPluginDirEntry->szName, rc, pfnVDCacheLoad));
2573 if (RT_SUCCESS(rc))
2574 rc = VERR_SYMBOL_NOT_FOUND;
2575 }
2576
2577 if (RT_SUCCESS(rc))
2578 {
2579 /* Get the function table. */
2580 rc = pfnVDCacheLoad(&pBackend);
2581 if (RT_SUCCESS(rc) && pBackend->cbSize == sizeof(VDCACHEBACKEND))
2582 {
2583 pBackend->hPlugin = hPlugin;
2584 vdAddCacheBackend(pBackend);
2585 }
2586 else
2587 LogFunc(("ignored plugin '%s': pBackend->cbSize=%d rc=%Rrc\n", pszPluginPath, pBackend->cbSize, rc));
2588 }
2589 else
2590 LogFunc(("ignored plugin '%s': rc=%Rrc\n", pszPluginPath, rc));
2591
2592 if (RT_FAILURE(rc))
2593 RTLdrClose(hPlugin);
2594 }
2595 RTStrFree(pszPluginPath);
2596 }
2597out:
2598 if (rc == VERR_NO_MORE_FILES)
2599 rc = VINF_SUCCESS;
2600 RTStrFree(pszPluginFilter);
2601 if (pPluginDirEntry)
2602 RTMemFree(pPluginDirEntry);
2603 if (pPluginDir)
2604 RTDirClose(pPluginDir);
2605 return rc;
2606#else
2607 return VINF_SUCCESS;
2608#endif
2609}
2610
2611/**
2612 * VD async I/O interface open callback.
2613 */
2614static int vdIOOpenFallback(void *pvUser, const char *pszLocation,
2615 uint32_t fOpen, PFNVDCOMPLETED pfnCompleted,
2616 void **ppStorage)
2617{
2618 PVDIIOFALLBACKSTORAGE pStorage = (PVDIIOFALLBACKSTORAGE)RTMemAllocZ(sizeof(VDIIOFALLBACKSTORAGE));
2619
2620 if (!pStorage)
2621 return VERR_NO_MEMORY;
2622
2623 pStorage->pfnCompleted = pfnCompleted;
2624
2625 /* Open the file. */
2626 int rc = RTFileOpen(&pStorage->File, pszLocation, fOpen);
2627 if (RT_SUCCESS(rc))
2628 {
2629 *ppStorage = pStorage;
2630 return VINF_SUCCESS;
2631 }
2632
2633 RTMemFree(pStorage);
2634 return rc;
2635}
2636
2637/**
2638 * VD async I/O interface close callback.
2639 */
2640static int vdIOCloseFallback(void *pvUser, void *pvStorage)
2641{
2642 PVDIIOFALLBACKSTORAGE pStorage = (PVDIIOFALLBACKSTORAGE)pvStorage;
2643
2644 RTFileClose(pStorage->File);
2645 RTMemFree(pStorage);
2646 return VINF_SUCCESS;
2647}
2648
2649static int vdIODeleteFallback(void *pvUser, const char *pcszFilename)
2650{
2651 return RTFileDelete(pcszFilename);
2652}
2653
2654static int vdIOMoveFallback(void *pvUser, const char *pcszSrc, const char *pcszDst, unsigned fMove)
2655{
2656 return RTFileMove(pcszSrc, pcszDst, fMove);
2657}
2658
2659static int vdIOGetFreeSpaceFallback(void *pvUser, const char *pcszFilename, int64_t *pcbFreeSpace)
2660{
2661 return RTFsQuerySizes(pcszFilename, NULL, pcbFreeSpace, NULL, NULL);
2662}
2663
2664static int vdIOGetModificationTimeFallback(void *pvUser, const char *pcszFilename, PRTTIMESPEC pModificationTime)
2665{
2666 RTFSOBJINFO info;
2667 int rc = RTPathQueryInfo(pcszFilename, &info, RTFSOBJATTRADD_NOTHING);
2668 if (RT_SUCCESS(rc))
2669 *pModificationTime = info.ModificationTime;
2670 return rc;
2671}
2672
2673/**
2674 * VD async I/O interface callback for retrieving the file size.
2675 */
2676static int vdIOGetSizeFallback(void *pvUser, void *pvStorage, uint64_t *pcbSize)
2677{
2678 PVDIIOFALLBACKSTORAGE pStorage = (PVDIIOFALLBACKSTORAGE)pvStorage;
2679
2680 return RTFileGetSize(pStorage->File, pcbSize);
2681}
2682
2683/**
2684 * VD async I/O interface callback for setting the file size.
2685 */
2686static int vdIOSetSizeFallback(void *pvUser, void *pvStorage, uint64_t cbSize)
2687{
2688 PVDIIOFALLBACKSTORAGE pStorage = (PVDIIOFALLBACKSTORAGE)pvStorage;
2689
2690 return RTFileSetSize(pStorage->File, cbSize);
2691}
2692
2693/**
2694 * VD async I/O interface callback for a synchronous write to the file.
2695 */
2696static int vdIOWriteSyncFallback(void *pvUser, void *pvStorage, uint64_t uOffset,
2697 const void *pvBuf, size_t cbWrite, size_t *pcbWritten)
2698{
2699 PVDIIOFALLBACKSTORAGE pStorage = (PVDIIOFALLBACKSTORAGE)pvStorage;
2700
2701 return RTFileWriteAt(pStorage->File, uOffset, pvBuf, cbWrite, pcbWritten);
2702}
2703
2704/**
2705 * VD async I/O interface callback for a synchronous read from the file.
2706 */
2707static int vdIOReadSyncFallback(void *pvUser, void *pvStorage, uint64_t uOffset,
2708 void *pvBuf, size_t cbRead, size_t *pcbRead)
2709{
2710 PVDIIOFALLBACKSTORAGE pStorage = (PVDIIOFALLBACKSTORAGE)pvStorage;
2711
2712 return RTFileReadAt(pStorage->File, uOffset, pvBuf, cbRead, pcbRead);
2713}
2714
2715/**
2716 * VD async I/O interface callback for a synchronous flush of the file data.
2717 */
2718static int vdIOFlushSyncFallback(void *pvUser, void *pvStorage)
2719{
2720 PVDIIOFALLBACKSTORAGE pStorage = (PVDIIOFALLBACKSTORAGE)pvStorage;
2721
2722 return RTFileFlush(pStorage->File);
2723}
2724
2725/**
2726 * VD async I/O interface callback for a asynchronous read from the file.
2727 */
2728static int vdIOReadAsyncFallback(void *pvUser, void *pStorage, uint64_t uOffset,
2729 PCRTSGSEG paSegments, size_t cSegments,
2730 size_t cbRead, void *pvCompletion,
2731 void **ppTask)
2732{
2733 return VERR_NOT_IMPLEMENTED;
2734}
2735
2736/**
2737 * VD async I/O interface callback for a asynchronous write to the file.
2738 */
2739static int vdIOWriteAsyncFallback(void *pvUser, void *pStorage, uint64_t uOffset,
2740 PCRTSGSEG paSegments, size_t cSegments,
2741 size_t cbWrite, void *pvCompletion,
2742 void **ppTask)
2743{
2744 return VERR_NOT_IMPLEMENTED;
2745}
2746
2747/**
2748 * VD async I/O interface callback for a asynchronous flush of the file data.
2749 */
2750static int vdIOFlushAsyncFallback(void *pvUser, void *pStorage,
2751 void *pvCompletion, void **ppTask)
2752{
2753 return VERR_NOT_IMPLEMENTED;
2754}
2755
2756/**
2757 * Internal - Continues an I/O context after
2758 * it was halted because of an active transfer.
2759 */
2760static int vdIoCtxContinue(PVDIOCTX pIoCtx, int rcReq)
2761{
2762 PVBOXHDD pDisk = pIoCtx->pDisk;
2763 int rc = VINF_SUCCESS;
2764
2765 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
2766
2767 if (RT_FAILURE(rcReq))
2768 ASMAtomicCmpXchgS32(&pIoCtx->rcReq, rcReq, VINF_SUCCESS);
2769
2770 if (!pIoCtx->fBlocked)
2771 {
2772 /* Continue the transfer */
2773 rc = vdIoCtxProcess(pIoCtx);
2774
2775 if ( rc == VINF_VD_ASYNC_IO_FINISHED
2776 && ASMAtomicCmpXchgBool(&pIoCtx->fComplete, true, false))
2777 {
2778 LogFlowFunc(("I/O context completed pIoCtx=%#p\n", pIoCtx));
2779 if (pIoCtx->pIoCtxParent)
2780 {
2781 PVDIOCTX pIoCtxParent = pIoCtx->pIoCtxParent;
2782
2783 Assert(!pIoCtxParent->pIoCtxParent);
2784 if (RT_FAILURE(pIoCtx->rcReq))
2785 ASMAtomicCmpXchgS32(&pIoCtxParent->rcReq, pIoCtx->rcReq, VINF_SUCCESS);
2786
2787 if (pIoCtx->enmTxDir == VDIOCTXTXDIR_WRITE)
2788 {
2789 LogFlowFunc(("I/O context transferred %u bytes for the parent pIoCtxParent=%p\n",
2790 pIoCtx->Type.Child.cbTransferParent, pIoCtxParent));
2791
2792 /* Update the parent state. */
2793 Assert(pIoCtxParent->cbTransferLeft >= pIoCtx->Type.Child.cbTransferParent);
2794 ASMAtomicSubU32(&pIoCtxParent->cbTransferLeft, pIoCtx->Type.Child.cbTransferParent);
2795 }
2796 else
2797 Assert(pIoCtx->enmTxDir == VDIOCTXTXDIR_FLUSH);
2798
2799 /*
2800 * A completed child write means that we finished growing the image.
2801 * We have to process any pending writes now.
2802 */
2803 vdIoCtxUnlockDisk(pDisk, pIoCtxParent, false /* fProcessDeferredReqs */);
2804
2805 /* Unblock the parent */
2806 pIoCtxParent->fBlocked = false;
2807
2808 rc = vdIoCtxProcess(pIoCtxParent);
2809
2810 if ( rc == VINF_VD_ASYNC_IO_FINISHED
2811 && ASMAtomicCmpXchgBool(&pIoCtxParent->fComplete, true, false))
2812 {
2813 RTCritSectLeave(&pDisk->CritSect);
2814 LogFlowFunc(("Parent I/O context completed pIoCtxParent=%#p rcReq=%Rrc\n", pIoCtxParent, pIoCtxParent->rcReq));
2815 pIoCtxParent->Type.Root.pfnComplete(pIoCtxParent->Type.Root.pvUser1,
2816 pIoCtxParent->Type.Root.pvUser2,
2817 pIoCtxParent->rcReq);
2818 vdThreadFinishWrite(pDisk);
2819 vdIoCtxFree(pDisk, pIoCtxParent);
2820 RTCritSectEnter(&pDisk->CritSect);
2821 }
2822
2823 /* Process any pending writes if the current request didn't caused another growing. */
2824 if ( !RTListIsEmpty(&pDisk->ListWriteLocked)
2825 && !vdIoCtxIsDiskLockOwner(pDisk, pIoCtx))
2826 {
2827 RTLISTNODE ListTmp;
2828
2829 LogFlowFunc(("Before: pNext=%#p pPrev=%#p\n", pDisk->ListWriteLocked.pNext,
2830 pDisk->ListWriteLocked.pPrev));
2831
2832 RTListMove(&ListTmp, &pDisk->ListWriteLocked);
2833
2834 LogFlowFunc(("After: pNext=%#p pPrev=%#p\n", pDisk->ListWriteLocked.pNext,
2835 pDisk->ListWriteLocked.pPrev));
2836
2837 RTCritSectLeave(&pDisk->CritSect);
2838
2839 /* Process the list. */
2840 do
2841 {
2842 PVDIOCTXDEFERRED pDeferred = RTListGetFirst(&ListTmp, VDIOCTXDEFERRED, NodeDeferred);
2843 PVDIOCTX pIoCtxWait = pDeferred->pIoCtx;
2844
2845 AssertPtr(pIoCtxWait);
2846
2847 RTListNodeRemove(&pDeferred->NodeDeferred);
2848 RTMemFree(pDeferred);
2849
2850 Assert(!pIoCtxWait->pIoCtxParent);
2851
2852 pIoCtxWait->fBlocked = false;
2853 LogFlowFunc(("Processing waiting I/O context pIoCtxWait=%#p\n", pIoCtxWait));
2854
2855 rc = vdIoCtxProcess(pIoCtxWait);
2856 if ( rc == VINF_VD_ASYNC_IO_FINISHED
2857 && ASMAtomicCmpXchgBool(&pIoCtxWait->fComplete, true, false))
2858 {
2859 LogFlowFunc(("Waiting I/O context completed pIoCtxWait=%#p\n", pIoCtxWait));
2860 vdThreadFinishWrite(pDisk);
2861 pIoCtxWait->Type.Root.pfnComplete(pIoCtxWait->Type.Root.pvUser1,
2862 pIoCtxWait->Type.Root.pvUser2,
2863 pIoCtxWait->rcReq);
2864 vdIoCtxFree(pDisk, pIoCtxWait);
2865 }
2866 } while (!RTListIsEmpty(&ListTmp));
2867
2868 RTCritSectEnter(&pDisk->CritSect);
2869 }
2870 }
2871 else
2872 {
2873 if (pIoCtx->enmTxDir == VDIOCTXTXDIR_FLUSH)
2874 {
2875 vdIoCtxUnlockDisk(pDisk, pIoCtx, true /* fProcessDerredReqs */);
2876 vdThreadFinishWrite(pDisk);
2877 }
2878 else if (pIoCtx->enmTxDir == VDIOCTXTXDIR_WRITE)
2879 vdThreadFinishWrite(pDisk);
2880 else
2881 {
2882 Assert(pIoCtx->enmTxDir == VDIOCTXTXDIR_READ);
2883 vdThreadFinishRead(pDisk);
2884 }
2885
2886 LogFlowFunc(("I/O context completed pIoCtx=%#p rcReq=%Rrc\n", pIoCtx, pIoCtx->rcReq));
2887 RTCritSectLeave(&pDisk->CritSect);
2888 pIoCtx->Type.Root.pfnComplete(pIoCtx->Type.Root.pvUser1,
2889 pIoCtx->Type.Root.pvUser2,
2890 pIoCtx->rcReq);
2891 RTCritSectEnter(&pDisk->CritSect);
2892 }
2893
2894 vdIoCtxFree(pDisk, pIoCtx);
2895 }
2896 }
2897
2898 return VINF_SUCCESS;
2899}
2900
2901/**
2902 * Internal - Called when user transfer completed.
2903 */
2904static int vdUserXferCompleted(PVDIOSTORAGE pIoStorage, PVDIOCTX pIoCtx,
2905 PFNVDXFERCOMPLETED pfnComplete, void *pvUser,
2906 size_t cbTransfer, int rcReq)
2907{
2908 int rc = VINF_SUCCESS;
2909 bool fIoCtxContinue = true;
2910 PVBOXHDD pDisk = pIoCtx->pDisk;
2911
2912 LogFlowFunc(("pIoStorage=%#p pIoCtx=%#p pfnComplete=%#p pvUser=%#p cbTransfer=%zu rcReq=%Rrc\n",
2913 pIoStorage, pIoCtx, pfnComplete, pvUser, cbTransfer, rcReq));
2914
2915 RTCritSectEnter(&pDisk->CritSect);
2916 Assert(pIoCtx->cbTransferLeft >= cbTransfer);
2917 ASMAtomicSubU32(&pIoCtx->cbTransferLeft, cbTransfer);
2918 ASMAtomicDecU32(&pIoCtx->cDataTransfersPending);
2919
2920 if (pfnComplete)
2921 rc = pfnComplete(pIoStorage->pVDIo->pBackendData, pIoCtx, pvUser, rcReq);
2922
2923 if (RT_SUCCESS(rc))
2924 rc = vdIoCtxContinue(pIoCtx, rcReq);
2925 else if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
2926 rc = VINF_SUCCESS;
2927
2928 RTCritSectLeave(&pDisk->CritSect);
2929
2930 return rc;
2931}
2932
2933/**
2934 * Internal - Called when a meta transfer completed.
2935 */
2936static int vdMetaXferCompleted(PVDIOSTORAGE pIoStorage, PFNVDXFERCOMPLETED pfnComplete, void *pvUser,
2937 PVDMETAXFER pMetaXfer, int rcReq)
2938{
2939 PVBOXHDD pDisk = pIoStorage->pVDIo->pDisk;
2940 RTLISTNODE ListIoCtxWaiting;
2941 bool fFlush;
2942
2943 LogFlowFunc(("pIoStorage=%#p pfnComplete=%#p pvUser=%#p pMetaXfer=%#p rcReq=%Rrc\n",
2944 pIoStorage, pfnComplete, pvUser, pMetaXfer, rcReq));
2945
2946 RTCritSectEnter(&pDisk->CritSect);
2947 fFlush = VDMETAXFER_TXDIR_GET(pMetaXfer->fFlags) == VDMETAXFER_TXDIR_FLUSH;
2948 VDMETAXFER_TXDIR_SET(pMetaXfer->fFlags, VDMETAXFER_TXDIR_NONE);
2949
2950 if (!fFlush)
2951 {
2952 RTListMove(&ListIoCtxWaiting, &pMetaXfer->ListIoCtxWaiting);
2953
2954 if (RT_FAILURE(rcReq))
2955 {
2956 /* Remove from the AVL tree. */
2957 LogFlow(("Removing meta xfer=%#p\n", pMetaXfer));
2958 bool fRemoved = RTAvlrFileOffsetRemove(pIoStorage->pTreeMetaXfers, pMetaXfer->Core.Key) != NULL;
2959 Assert(fRemoved);
2960 RTMemFree(pMetaXfer);
2961 }
2962 else
2963 {
2964 /* Increase the reference counter to make sure it doesn't go away before the last context is processed. */
2965 pMetaXfer->cRefs++;
2966 }
2967 }
2968 else
2969 RTListMove(&ListIoCtxWaiting, &pMetaXfer->ListIoCtxWaiting);
2970
2971 /* Go through the waiting list and continue the I/O contexts. */
2972 while (!RTListIsEmpty(&ListIoCtxWaiting))
2973 {
2974 int rc = VINF_SUCCESS;
2975 bool fContinue = true;
2976 PVDIOCTXDEFERRED pDeferred = RTListGetFirst(&ListIoCtxWaiting, VDIOCTXDEFERRED, NodeDeferred);
2977 PVDIOCTX pIoCtx = pDeferred->pIoCtx;
2978 RTListNodeRemove(&pDeferred->NodeDeferred);
2979
2980 RTMemFree(pDeferred);
2981 ASMAtomicDecU32(&pIoCtx->cMetaTransfersPending);
2982
2983 if (pfnComplete)
2984 rc = pfnComplete(pIoStorage->pVDIo->pBackendData, pIoCtx, pvUser, rcReq);
2985
2986 LogFlow(("Completion callback for I/O context %#p returned %Rrc\n", pIoCtx, rc));
2987
2988 if (RT_SUCCESS(rc))
2989 {
2990 rc = vdIoCtxContinue(pIoCtx, rcReq);
2991 AssertRC(rc);
2992 }
2993 else
2994 Assert(rc == VERR_VD_ASYNC_IO_IN_PROGRESS);
2995 }
2996
2997 /* Remove if not used anymore. */
2998 if (RT_SUCCESS(rcReq) && !fFlush)
2999 {
3000 pMetaXfer->cRefs--;
3001 if (!pMetaXfer->cRefs && RTListIsEmpty(&pMetaXfer->ListIoCtxWaiting))
3002 {
3003 /* Remove from the AVL tree. */
3004 LogFlow(("Removing meta xfer=%#p\n", pMetaXfer));
3005 bool fRemoved = RTAvlrFileOffsetRemove(pIoStorage->pTreeMetaXfers, pMetaXfer->Core.Key) != NULL;
3006 Assert(fRemoved);
3007 RTMemFree(pMetaXfer);
3008 }
3009 }
3010 else if (fFlush)
3011 RTMemFree(pMetaXfer);
3012
3013 RTCritSectLeave(&pDisk->CritSect);
3014
3015 return VINF_SUCCESS;
3016}
3017
3018static int vdIOIntReqCompleted(void *pvUser, int rcReq)
3019{
3020 int rc = VINF_SUCCESS;
3021 PVDIOTASK pIoTask = (PVDIOTASK)pvUser;
3022 PVDIOSTORAGE pIoStorage = pIoTask->pIoStorage;
3023
3024 LogFlowFunc(("Task completed pIoTask=%#p\n", pIoTask));
3025
3026 if (!pIoTask->fMeta)
3027 rc = vdUserXferCompleted(pIoStorage, pIoTask->Type.User.pIoCtx,
3028 pIoTask->pfnComplete, pIoTask->pvUser,
3029 pIoTask->Type.User.cbTransfer, rcReq);
3030 else
3031 rc = vdMetaXferCompleted(pIoStorage, pIoTask->pfnComplete, pIoTask->pvUser,
3032 pIoTask->Type.Meta.pMetaXfer, rcReq);
3033
3034 vdIoTaskFree(pIoStorage->pVDIo->pDisk, pIoTask);
3035
3036 return rc;
3037}
3038
3039/**
3040 * VD I/O interface callback for opening a file.
3041 */
3042static int vdIOIntOpen(void *pvUser, const char *pszLocation,
3043 unsigned uOpenFlags, PPVDIOSTORAGE ppIoStorage)
3044{
3045 int rc = VINF_SUCCESS;
3046 PVDIO pVDIo = (PVDIO)pvUser;
3047 PVDIOSTORAGE pIoStorage = (PVDIOSTORAGE)RTMemAllocZ(sizeof(VDIOSTORAGE));
3048
3049 if (!pIoStorage)
3050 return VERR_NO_MEMORY;
3051
3052 /* Create the AVl tree. */
3053 pIoStorage->pTreeMetaXfers = (PAVLRFOFFTREE)RTMemAllocZ(sizeof(AVLRFOFFTREE));
3054 if (pIoStorage->pTreeMetaXfers)
3055 {
3056 rc = pVDIo->pInterfaceIOCallbacks->pfnOpen(pVDIo->pInterfaceIO->pvUser,
3057 pszLocation, uOpenFlags,
3058 vdIOIntReqCompleted,
3059 &pIoStorage->pStorage);
3060 if (RT_SUCCESS(rc))
3061 {
3062 pIoStorage->pVDIo = pVDIo;
3063 *ppIoStorage = pIoStorage;
3064 return VINF_SUCCESS;
3065 }
3066
3067 RTMemFree(pIoStorage->pTreeMetaXfers);
3068 }
3069 else
3070 rc = VERR_NO_MEMORY;
3071
3072 RTMemFree(pIoStorage);
3073 return rc;
3074}
3075
3076static int vdIOIntTreeMetaXferDestroy(PAVLRFOFFNODECORE pNode, void *pvUser)
3077{
3078 AssertMsgFailed(("Tree should be empty at this point!\n"));
3079 return VINF_SUCCESS;
3080}
3081
3082static int vdIOIntClose(void *pvUser, PVDIOSTORAGE pIoStorage)
3083{
3084 PVDIO pVDIo = (PVDIO)pvUser;
3085
3086 int rc = pVDIo->pInterfaceIOCallbacks->pfnClose(pVDIo->pInterfaceIO->pvUser,
3087 pIoStorage->pStorage);
3088 AssertRC(rc);
3089
3090 RTAvlrFileOffsetDestroy(pIoStorage->pTreeMetaXfers, vdIOIntTreeMetaXferDestroy, NULL);
3091 RTMemFree(pIoStorage->pTreeMetaXfers);
3092 RTMemFree(pIoStorage);
3093 return VINF_SUCCESS;
3094}
3095
3096static int vdIOIntDelete(void *pvUser, const char *pcszFilename)
3097{
3098 PVDIO pVDIo = (PVDIO)pvUser;
3099 return pVDIo->pInterfaceIOCallbacks->pfnDelete(pVDIo->pInterfaceIO->pvUser,
3100 pcszFilename);
3101}
3102
3103static int vdIOIntMove(void *pvUser, const char *pcszSrc, const char *pcszDst,
3104 unsigned fMove)
3105{
3106 PVDIO pVDIo = (PVDIO)pvUser;
3107 return pVDIo->pInterfaceIOCallbacks->pfnMove(pVDIo->pInterfaceIO->pvUser,
3108 pcszSrc, pcszDst, fMove);
3109}
3110
3111static int vdIOIntGetFreeSpace(void *pvUser, const char *pcszFilename,
3112 int64_t *pcbFreeSpace)
3113{
3114 PVDIO pVDIo = (PVDIO)pvUser;
3115 return pVDIo->pInterfaceIOCallbacks->pfnGetFreeSpace(pVDIo->pInterfaceIO->pvUser,
3116 pcszFilename,
3117 pcbFreeSpace);
3118}
3119
3120static int vdIOIntGetModificationTime(void *pvUser, const char *pcszFilename,
3121 PRTTIMESPEC pModificationTime)
3122{
3123 PVDIO pVDIo = (PVDIO)pvUser;
3124 return pVDIo->pInterfaceIOCallbacks->pfnGetModificationTime(pVDIo->pInterfaceIO->pvUser,
3125 pcszFilename,
3126 pModificationTime);
3127}
3128
3129static int vdIOIntGetSize(void *pvUser, PVDIOSTORAGE pIoStorage,
3130 uint64_t *pcbSize)
3131{
3132 PVDIO pVDIo = (PVDIO)pvUser;
3133 return pVDIo->pInterfaceIOCallbacks->pfnGetSize(pVDIo->pInterfaceIO->pvUser,
3134 pIoStorage->pStorage,
3135 pcbSize);
3136}
3137
3138static int vdIOIntSetSize(void *pvUser, PVDIOSTORAGE pIoStorage,
3139 uint64_t cbSize)
3140{
3141 PVDIO pVDIo = (PVDIO)pvUser;
3142
3143 return pVDIo->pInterfaceIOCallbacks->pfnSetSize(pVDIo->pInterfaceIO->pvUser,
3144 pIoStorage->pStorage,
3145 cbSize);
3146}
3147
3148static int vdIOIntWriteSync(void *pvUser, PVDIOSTORAGE pIoStorage,
3149 uint64_t uOffset, const void *pvBuf,
3150 size_t cbWrite, size_t *pcbWritten)
3151{
3152 PVDIO pVDIo = (PVDIO)pvUser;
3153
3154 return pVDIo->pInterfaceIOCallbacks->pfnWriteSync(pVDIo->pInterfaceIO->pvUser,
3155 pIoStorage->pStorage,
3156 uOffset, pvBuf, cbWrite,
3157 pcbWritten);
3158}
3159
3160static int vdIOIntReadSync(void *pvUser, PVDIOSTORAGE pIoStorage,
3161 uint64_t uOffset, void *pvBuf, size_t cbRead,
3162 size_t *pcbRead)
3163{
3164 PVDIO pVDIo = (PVDIO)pvUser;
3165 return pVDIo->pInterfaceIOCallbacks->pfnReadSync(pVDIo->pInterfaceIO->pvUser,
3166 pIoStorage->pStorage,
3167 uOffset, pvBuf, cbRead,
3168 pcbRead);
3169}
3170
3171static int vdIOIntFlushSync(void *pvUser, PVDIOSTORAGE pIoStorage)
3172{
3173 PVDIO pVDIo = (PVDIO)pvUser;
3174 return pVDIo->pInterfaceIOCallbacks->pfnFlushSync(pVDIo->pInterfaceIO->pvUser,
3175 pIoStorage->pStorage);
3176}
3177
3178static int vdIOIntReadUserAsync(void *pvUser, PVDIOSTORAGE pIoStorage,
3179 uint64_t uOffset, PVDIOCTX pIoCtx,
3180 size_t cbRead)
3181{
3182 int rc = VINF_SUCCESS;
3183 PVDIO pVDIo = (PVDIO)pvUser;
3184 PVBOXHDD pDisk = pVDIo->pDisk;
3185
3186 LogFlowFunc(("pvUser=%#p pIoStorage=%#p uOffset=%llu pIoCtx=%#p cbRead=%u\n",
3187 pvUser, pIoStorage, uOffset, pIoCtx, cbRead));
3188
3189 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3190
3191 Assert(cbRead > 0);
3192
3193 /* Build the S/G array and spawn a new I/O task */
3194 while (cbRead)
3195 {
3196 RTSGSEG aSeg[VD_IO_TASK_SEGMENTS_MAX];
3197 unsigned cSegments = VD_IO_TASK_SEGMENTS_MAX;
3198 size_t cbTaskRead = 0;
3199
3200 cbTaskRead = RTSgBufSegArrayCreate(&pIoCtx->SgBuf, aSeg, &cSegments, cbRead);
3201
3202 Assert(cSegments > 0);
3203 Assert(cbTaskRead > 0);
3204 AssertMsg(cbTaskRead <= cbRead, ("Invalid number of bytes to read\n"));
3205
3206 LogFlow(("Reading %u bytes into %u segments\n", cbTaskRead, cSegments));
3207
3208#ifdef RT_STRICT
3209 for (unsigned i = 0; i < cSegments; i++)
3210 AssertMsg(aSeg[i].pvSeg && !(aSeg[i].cbSeg % 512),
3211 ("Segment %u is invalid\n", i));
3212#endif
3213
3214 PVDIOTASK pIoTask = vdIoTaskUserAlloc(pIoStorage, NULL, NULL, pIoCtx, cbTaskRead);
3215
3216 if (!pIoTask)
3217 return VERR_NO_MEMORY;
3218
3219 ASMAtomicIncU32(&pIoCtx->cDataTransfersPending);
3220
3221 void *pvTask;
3222 rc = pVDIo->pInterfaceIOCallbacks->pfnReadAsync(pVDIo->pInterfaceIO->pvUser,
3223 pIoStorage->pStorage,
3224 uOffset, aSeg, cSegments,
3225 cbTaskRead, pIoTask,
3226 &pvTask);
3227 if (RT_SUCCESS(rc))
3228 {
3229 AssertMsg(cbTaskRead <= pIoCtx->cbTransferLeft, ("Impossible!\n"));
3230 ASMAtomicSubU32(&pIoCtx->cbTransferLeft, cbTaskRead);
3231 ASMAtomicDecU32(&pIoCtx->cDataTransfersPending);
3232 vdIoTaskFree(pDisk, pIoTask);
3233 }
3234 else if (rc != VERR_VD_ASYNC_IO_IN_PROGRESS)
3235 {
3236 ASMAtomicDecU32(&pIoCtx->cDataTransfersPending);
3237 vdIoTaskFree(pDisk, pIoTask);
3238 break;
3239 }
3240
3241 uOffset += cbTaskRead;
3242 cbRead -= cbTaskRead;
3243 }
3244
3245 LogFlowFunc(("returns rc=%Rrc\n", rc));
3246 return rc;
3247}
3248
3249static int vdIOIntWriteUserAsync(void *pvUser, PVDIOSTORAGE pIoStorage,
3250 uint64_t uOffset, PVDIOCTX pIoCtx,
3251 size_t cbWrite,
3252 PFNVDXFERCOMPLETED pfnComplete,
3253 void *pvCompleteUser)
3254{
3255 int rc = VINF_SUCCESS;
3256 PVDIO pVDIo = (PVDIO)pvUser;
3257 PVBOXHDD pDisk = pVDIo->pDisk;
3258
3259 LogFlowFunc(("pvUser=%#p pIoStorage=%#p uOffset=%llu pIoCtx=%#p cbWrite=%u\n",
3260 pvUser, pIoStorage, uOffset, pIoCtx, cbWrite));
3261
3262 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3263
3264 Assert(cbWrite > 0);
3265
3266 /* Build the S/G array and spawn a new I/O task */
3267 while (cbWrite)
3268 {
3269 RTSGSEG aSeg[VD_IO_TASK_SEGMENTS_MAX];
3270 unsigned cSegments = VD_IO_TASK_SEGMENTS_MAX;
3271 size_t cbTaskWrite = 0;
3272
3273 cbTaskWrite = RTSgBufSegArrayCreate(&pIoCtx->SgBuf, aSeg, &cSegments, cbWrite);
3274
3275 Assert(cSegments > 0);
3276 Assert(cbTaskWrite > 0);
3277 AssertMsg(cbTaskWrite <= cbWrite, ("Invalid number of bytes to write\n"));
3278
3279 LogFlow(("Writing %u bytes from %u segments\n", cbTaskWrite, cSegments));
3280
3281#ifdef DEBUG
3282 for (unsigned i = 0; i < cSegments; i++)
3283 AssertMsg(aSeg[i].pvSeg && !(aSeg[i].cbSeg % 512),
3284 ("Segment %u is invalid\n", i));
3285#endif
3286
3287 PVDIOTASK pIoTask = vdIoTaskUserAlloc(pIoStorage, pfnComplete, pvCompleteUser, pIoCtx, cbTaskWrite);
3288
3289 if (!pIoTask)
3290 return VERR_NO_MEMORY;
3291
3292 ASMAtomicIncU32(&pIoCtx->cDataTransfersPending);
3293
3294 void *pvTask;
3295 rc = pVDIo->pInterfaceIOCallbacks->pfnWriteAsync(pVDIo->pInterfaceIO->pvUser,
3296 pIoStorage->pStorage,
3297 uOffset, aSeg, cSegments,
3298 cbTaskWrite, pIoTask,
3299 &pvTask);
3300 if (RT_SUCCESS(rc))
3301 {
3302 AssertMsg(cbTaskWrite <= pIoCtx->cbTransferLeft, ("Impossible!\n"));
3303 ASMAtomicSubU32(&pIoCtx->cbTransferLeft, cbTaskWrite);
3304 ASMAtomicDecU32(&pIoCtx->cDataTransfersPending);
3305 vdIoTaskFree(pDisk, pIoTask);
3306 }
3307 else if (rc != VERR_VD_ASYNC_IO_IN_PROGRESS)
3308 {
3309 ASMAtomicDecU32(&pIoCtx->cDataTransfersPending);
3310 vdIoTaskFree(pDisk, pIoTask);
3311 break;
3312 }
3313
3314 uOffset += cbTaskWrite;
3315 cbWrite -= cbTaskWrite;
3316 }
3317
3318 return rc;
3319}
3320
3321static int vdIOIntReadMetaAsync(void *pvUser, PVDIOSTORAGE pIoStorage,
3322 uint64_t uOffset, void *pvBuf,
3323 size_t cbRead, PVDIOCTX pIoCtx,
3324 PPVDMETAXFER ppMetaXfer,
3325 PFNVDXFERCOMPLETED pfnComplete,
3326 void *pvCompleteUser)
3327{
3328 PVDIO pVDIo = (PVDIO)pvUser;
3329 PVBOXHDD pDisk = pVDIo->pDisk;
3330 int rc = VINF_SUCCESS;
3331 RTSGSEG Seg;
3332 PVDIOTASK pIoTask;
3333 PVDMETAXFER pMetaXfer = NULL;
3334 void *pvTask = NULL;
3335
3336 LogFlowFunc(("pvUser=%#p pIoStorage=%#p uOffset=%llu pvBuf=%#p cbRead=%u\n",
3337 pvUser, pIoStorage, uOffset, pvBuf, cbRead));
3338
3339 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3340
3341 pMetaXfer = (PVDMETAXFER)RTAvlrFileOffsetGet(pIoStorage->pTreeMetaXfers, uOffset);
3342 if (!pMetaXfer)
3343 {
3344#ifdef RT_STRICT
3345 pMetaXfer = (PVDMETAXFER)RTAvlrFileOffsetGetBestFit(pIoStorage->pTreeMetaXfers, uOffset, false /* fAbove */);
3346 AssertMsg(!pMetaXfer || (pMetaXfer->Core.Key + (RTFOFF)pMetaXfer->cbMeta <= (RTFOFF)uOffset),
3347 ("Overlapping meta transfers!\n"));
3348#endif
3349
3350 /* Allocate a new meta transfer. */
3351 pMetaXfer = vdMetaXferAlloc(pIoStorage, uOffset, cbRead);
3352 if (!pMetaXfer)
3353 return VERR_NO_MEMORY;
3354
3355 pIoTask = vdIoTaskMetaAlloc(pIoStorage, pfnComplete, pvCompleteUser, pMetaXfer);
3356 if (!pIoTask)
3357 {
3358 RTMemFree(pMetaXfer);
3359 return VERR_NO_MEMORY;
3360 }
3361
3362 Seg.cbSeg = cbRead;
3363 Seg.pvSeg = pMetaXfer->abData;
3364
3365 VDMETAXFER_TXDIR_SET(pMetaXfer->fFlags, VDMETAXFER_TXDIR_READ);
3366 rc = pVDIo->pInterfaceIOCallbacks->pfnReadAsync(pVDIo->pInterfaceIO->pvUser,
3367 pIoStorage->pStorage,
3368 uOffset, &Seg, 1,
3369 cbRead, pIoTask,
3370 &pvTask);
3371
3372 if (RT_SUCCESS(rc) || rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
3373 {
3374 bool fInserted = RTAvlrFileOffsetInsert(pIoStorage->pTreeMetaXfers, &pMetaXfer->Core);
3375 Assert(fInserted);
3376 }
3377 else
3378 RTMemFree(pMetaXfer);
3379
3380 if (RT_SUCCESS(rc))
3381 {
3382 VDMETAXFER_TXDIR_SET(pMetaXfer->fFlags, VDMETAXFER_TXDIR_NONE);
3383 vdIoTaskFree(pDisk, pIoTask);
3384 }
3385 else if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS && !pfnComplete)
3386 rc = VERR_VD_NOT_ENOUGH_METADATA;
3387 }
3388
3389 Assert(VALID_PTR(pMetaXfer) || RT_FAILURE(rc));
3390
3391 if (RT_SUCCESS(rc) || rc == VERR_VD_NOT_ENOUGH_METADATA || rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
3392 {
3393 /* If it is pending add the request to the list. */
3394 if (VDMETAXFER_TXDIR_GET(pMetaXfer->fFlags) == VDMETAXFER_TXDIR_READ)
3395 {
3396 PVDIOCTXDEFERRED pDeferred = (PVDIOCTXDEFERRED)RTMemAllocZ(sizeof(VDIOCTXDEFERRED));
3397 AssertPtr(pDeferred);
3398
3399 RTListInit(&pDeferred->NodeDeferred);
3400 pDeferred->pIoCtx = pIoCtx;
3401
3402 ASMAtomicIncU32(&pIoCtx->cMetaTransfersPending);
3403 RTListAppend(&pMetaXfer->ListIoCtxWaiting, &pDeferred->NodeDeferred);
3404 rc = VERR_VD_NOT_ENOUGH_METADATA;
3405 }
3406 else
3407 {
3408 /* Transfer the data. */
3409 pMetaXfer->cRefs++;
3410 Assert(pMetaXfer->cbMeta >= cbRead);
3411 Assert(pMetaXfer->Core.Key == (RTFOFF)uOffset);
3412 memcpy(pvBuf, pMetaXfer->abData, cbRead);
3413 *ppMetaXfer = pMetaXfer;
3414 }
3415 }
3416
3417 return rc;
3418}
3419
3420static int vdIOIntWriteMetaAsync(void *pvUser, PVDIOSTORAGE pIoStorage,
3421 uint64_t uOffset, void *pvBuf,
3422 size_t cbWrite, PVDIOCTX pIoCtx,
3423 PFNVDXFERCOMPLETED pfnComplete,
3424 void *pvCompleteUser)
3425{
3426 PVDIO pVDIo = (PVDIO)pvUser;
3427 PVBOXHDD pDisk = pVDIo->pDisk;
3428 int rc = VINF_SUCCESS;
3429 RTSGSEG Seg;
3430 PVDIOTASK pIoTask;
3431 PVDMETAXFER pMetaXfer = NULL;
3432 bool fInTree = false;
3433 void *pvTask = NULL;
3434
3435 LogFlowFunc(("pvUser=%#p pIoStorage=%#p uOffset=%llu pvBuf=%#p cbWrite=%u\n",
3436 pvUser, pIoStorage, uOffset, pvBuf, cbWrite));
3437
3438 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3439
3440 pMetaXfer = (PVDMETAXFER)RTAvlrFileOffsetGet(pIoStorage->pTreeMetaXfers, uOffset);
3441 if (!pMetaXfer)
3442 {
3443 /* Allocate a new meta transfer. */
3444 pMetaXfer = vdMetaXferAlloc(pIoStorage, uOffset, cbWrite);
3445 if (!pMetaXfer)
3446 return VERR_NO_MEMORY;
3447 }
3448 else
3449 {
3450 Assert(pMetaXfer->cbMeta >= cbWrite);
3451 Assert(pMetaXfer->Core.Key == (RTFOFF)uOffset);
3452 fInTree = true;
3453 }
3454
3455 Assert(VDMETAXFER_TXDIR_GET(pMetaXfer->fFlags) == VDMETAXFER_TXDIR_NONE);
3456
3457 pIoTask = vdIoTaskMetaAlloc(pIoStorage, pfnComplete, pvCompleteUser, pMetaXfer);
3458 if (!pIoTask)
3459 {
3460 RTMemFree(pMetaXfer);
3461 return VERR_NO_MEMORY;
3462 }
3463
3464 memcpy(pMetaXfer->abData, pvBuf, cbWrite);
3465 Seg.cbSeg = cbWrite;
3466 Seg.pvSeg = pMetaXfer->abData;
3467
3468 ASMAtomicIncU32(&pIoCtx->cMetaTransfersPending);
3469
3470 VDMETAXFER_TXDIR_SET(pMetaXfer->fFlags, VDMETAXFER_TXDIR_WRITE);
3471 rc = pVDIo->pInterfaceIOCallbacks->pfnWriteAsync(pVDIo->pInterfaceIO->pvUser,
3472 pIoStorage->pStorage,
3473 uOffset, &Seg, 1,
3474 cbWrite, pIoTask,
3475 &pvTask);
3476 if (RT_SUCCESS(rc))
3477 {
3478 VDMETAXFER_TXDIR_SET(pMetaXfer->fFlags, VDMETAXFER_TXDIR_NONE);
3479 ASMAtomicDecU32(&pIoCtx->cMetaTransfersPending);
3480 vdIoTaskFree(pDisk, pIoTask);
3481 if (fInTree && !pMetaXfer->cRefs)
3482 {
3483 LogFlow(("Removing meta xfer=%#p\n", pMetaXfer));
3484 bool fRemoved = RTAvlrFileOffsetRemove(pIoStorage->pTreeMetaXfers, pMetaXfer->Core.Key) != NULL;
3485 AssertMsg(fRemoved, ("Metadata transfer wasn't removed\n"));
3486 RTMemFree(pMetaXfer);
3487 pMetaXfer = NULL;
3488 }
3489 }
3490 else if (rc == VERR_VD_ASYNC_IO_IN_PROGRESS)
3491 {
3492 PVDIOCTXDEFERRED pDeferred = (PVDIOCTXDEFERRED)RTMemAllocZ(sizeof(VDIOCTXDEFERRED));
3493 AssertPtr(pDeferred);
3494
3495 RTListInit(&pDeferred->NodeDeferred);
3496 pDeferred->pIoCtx = pIoCtx;
3497
3498 if (!fInTree)
3499 {
3500 bool fInserted = RTAvlrFileOffsetInsert(pIoStorage->pTreeMetaXfers, &pMetaXfer->Core);
3501 Assert(fInserted);
3502 }
3503
3504 RTListAppend(&pMetaXfer->ListIoCtxWaiting, &pDeferred->NodeDeferred);
3505 }
3506 else
3507 {
3508 RTMemFree(pMetaXfer);
3509 pMetaXfer = NULL;
3510 }
3511
3512 return rc;
3513}
3514
3515static void vdIOIntMetaXferRelease(void *pvUser, PVDMETAXFER pMetaXfer)
3516{
3517 PVDIO pVDIo = (PVDIO)pvUser;
3518 PVBOXHDD pDisk = pVDIo->pDisk;
3519 PVDIOSTORAGE pIoStorage = pMetaXfer->pIoStorage;
3520
3521 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3522
3523 Assert( VDMETAXFER_TXDIR_GET(pMetaXfer->fFlags) == VDMETAXFER_TXDIR_NONE
3524 || VDMETAXFER_TXDIR_GET(pMetaXfer->fFlags) == VDMETAXFER_TXDIR_WRITE);
3525 Assert(pMetaXfer->cRefs > 0);
3526
3527 pMetaXfer->cRefs--;
3528 if ( !pMetaXfer->cRefs
3529 && RTListIsEmpty(&pMetaXfer->ListIoCtxWaiting)
3530 && VDMETAXFER_TXDIR_GET(pMetaXfer->fFlags) == VDMETAXFER_TXDIR_NONE)
3531 {
3532 /* Free the meta data entry. */
3533 LogFlow(("Removing meta xfer=%#p\n", pMetaXfer));
3534 bool fRemoved = RTAvlrFileOffsetRemove(pIoStorage->pTreeMetaXfers, pMetaXfer->Core.Key) != NULL;
3535 AssertMsg(fRemoved, ("Metadata transfer wasn't removed\n"));
3536
3537 RTMemFree(pMetaXfer);
3538 }
3539}
3540
3541static int vdIOIntFlushAsync(void *pvUser, PVDIOSTORAGE pIoStorage,
3542 PVDIOCTX pIoCtx, PFNVDXFERCOMPLETED pfnComplete,
3543 void *pvCompleteUser)
3544{
3545 PVDIO pVDIo = (PVDIO)pvUser;
3546 PVBOXHDD pDisk = pVDIo->pDisk;
3547 int rc = VINF_SUCCESS;
3548 PVDIOTASK pIoTask;
3549 PVDMETAXFER pMetaXfer = NULL;
3550 void *pvTask = NULL;
3551
3552 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3553
3554 LogFlowFunc(("pvUser=%#p pIoStorage=%#p pIoCtx=%#p\n",
3555 pvUser, pIoStorage, pIoCtx));
3556
3557 /* Allocate a new meta transfer. */
3558 pMetaXfer = vdMetaXferAlloc(pIoStorage, 0, 0);
3559 if (!pMetaXfer)
3560 return VERR_NO_MEMORY;
3561
3562 pIoTask = vdIoTaskMetaAlloc(pIoStorage, pfnComplete, pvUser, pMetaXfer);
3563 if (!pIoTask)
3564 {
3565 RTMemFree(pMetaXfer);
3566 return VERR_NO_MEMORY;
3567 }
3568
3569 ASMAtomicIncU32(&pIoCtx->cMetaTransfersPending);
3570
3571 PVDIOCTXDEFERRED pDeferred = (PVDIOCTXDEFERRED)RTMemAllocZ(sizeof(VDIOCTXDEFERRED));
3572 AssertPtr(pDeferred);
3573
3574 RTListInit(&pDeferred->NodeDeferred);
3575 pDeferred->pIoCtx = pIoCtx;
3576
3577 RTListAppend(&pMetaXfer->ListIoCtxWaiting, &pDeferred->NodeDeferred);
3578 VDMETAXFER_TXDIR_SET(pMetaXfer->fFlags, VDMETAXFER_TXDIR_FLUSH);
3579 rc = pVDIo->pInterfaceIOCallbacks->pfnFlushAsync(pVDIo->pInterfaceIO->pvUser,
3580 pIoStorage->pStorage,
3581 pIoTask, &pvTask);
3582 if (RT_SUCCESS(rc))
3583 {
3584 VDMETAXFER_TXDIR_SET(pMetaXfer->fFlags, VDMETAXFER_TXDIR_NONE);
3585 ASMAtomicDecU32(&pIoCtx->cMetaTransfersPending);
3586 vdIoTaskFree(pDisk, pIoTask);
3587 RTMemFree(pDeferred);
3588 RTMemFree(pMetaXfer);
3589 }
3590 else if (rc != VERR_VD_ASYNC_IO_IN_PROGRESS)
3591 RTMemFree(pMetaXfer);
3592
3593 return rc;
3594}
3595
3596static size_t vdIOIntIoCtxCopyTo(void *pvUser, PVDIOCTX pIoCtx,
3597 void *pvBuf, size_t cbBuf)
3598{
3599 PVDIO pVDIo = (PVDIO)pvUser;
3600 PVBOXHDD pDisk = pVDIo->pDisk;
3601 size_t cbCopied = 0;
3602
3603 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3604
3605 cbCopied = vdIoCtxCopyTo(pIoCtx, (uint8_t *)pvBuf, cbBuf);
3606 Assert(cbCopied == cbBuf);
3607
3608 ASMAtomicSubU32(&pIoCtx->cbTransferLeft, cbCopied);
3609
3610 return cbCopied;
3611}
3612
3613static size_t vdIOIntIoCtxCopyFrom(void *pvUser, PVDIOCTX pIoCtx,
3614 void *pvBuf, size_t cbBuf)
3615{
3616 PVDIO pVDIo = (PVDIO)pvUser;
3617 PVBOXHDD pDisk = pVDIo->pDisk;
3618 size_t cbCopied = 0;
3619
3620 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3621
3622 cbCopied = vdIoCtxCopyFrom(pIoCtx, (uint8_t *)pvBuf, cbBuf);
3623 Assert(cbCopied == cbBuf);
3624
3625 ASMAtomicSubU32(&pIoCtx->cbTransferLeft, cbCopied);
3626
3627 return cbCopied;
3628}
3629
3630static size_t vdIOIntIoCtxSet(void *pvUser, PVDIOCTX pIoCtx, int ch, size_t cb)
3631{
3632 PVDIO pVDIo = (PVDIO)pvUser;
3633 PVBOXHDD pDisk = pVDIo->pDisk;
3634 size_t cbSet = 0;
3635
3636 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3637
3638 cbSet = vdIoCtxSet(pIoCtx, ch, cb);
3639 Assert(cbSet == cb);
3640
3641 ASMAtomicSubU32(&pIoCtx->cbTransferLeft, cbSet);
3642
3643 return cbSet;
3644}
3645
3646static size_t vdIOIntIoCtxSegArrayCreate(void *pvUser, PVDIOCTX pIoCtx,
3647 PRTSGSEG paSeg, unsigned *pcSeg,
3648 size_t cbData)
3649{
3650 PVDIO pVDIo = (PVDIO)pvUser;
3651 PVBOXHDD pDisk = pVDIo->pDisk;
3652 size_t cbCreated = 0;
3653
3654 VD_THREAD_IS_CRITSECT_OWNER(pDisk);
3655
3656 cbCreated = RTSgBufSegArrayCreate(&pIoCtx->SgBuf, paSeg, pcSeg, cbData);
3657 Assert(!paSeg || cbData == cbCreated);
3658
3659 return cbCreated;
3660}
3661
3662static void vdIOIntIoCtxCompleted(void *pvUser, PVDIOCTX pIoCtx, int rcReq,
3663 size_t cbCompleted)
3664{
3665 PVDIO pVDIo = (PVDIO)pvUser;
3666 PVBOXHDD pDisk = pVDIo->pDisk;
3667
3668 /*
3669 * Grab the disk critical section to avoid races with other threads which
3670 * might still modify the I/O context.
3671 * Example is that iSCSI is doing an asynchronous write but calls us already
3672 * while the other thread is still hanging in vdWriteHelperAsync and couldn't update
3673 * the fBlocked state yet.
3674 * It can overwrite the state to true before we call vdIoCtxContinue and the
3675 * the request would hang indefinite.
3676 */
3677 int rc = RTCritSectEnter(&pDisk->CritSect);
3678 AssertRC(rc);
3679
3680 /* Continue */
3681 pIoCtx->fBlocked = false;
3682 ASMAtomicSubU32(&pIoCtx->cbTransferLeft, cbCompleted);
3683
3684 /* Clear the pointer to next transfer function in case we have nothing to transfer anymore.
3685 * @todo: Find a better way to prevent vdIoCtxContinue from calling the read/write helper again. */
3686 if (!pIoCtx->cbTransferLeft)
3687 pIoCtx->pfnIoCtxTransfer = NULL;
3688
3689 vdIoCtxContinue(pIoCtx, rcReq);
3690
3691 rc = RTCritSectLeave(&pDisk->CritSect);
3692 AssertRC(rc);
3693}
3694
3695/**
3696 * VD I/O interface callback for opening a file (limited version for VDGetFormat).
3697 */
3698static int vdIOIntOpenLimited(void *pvUser, const char *pszLocation,
3699 uint32_t fOpen, PPVDIOSTORAGE ppIoStorage)
3700{
3701 int rc = VINF_SUCCESS;
3702 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3703 PVDIOSTORAGE pIoStorage = (PVDIOSTORAGE)RTMemAllocZ(sizeof(VDIOSTORAGE));
3704
3705 if (!pIoStorage)
3706 return VERR_NO_MEMORY;
3707
3708 rc = pInterfaceIOCallbacks->pfnOpen(NULL, pszLocation, fOpen,
3709 NULL, &pIoStorage->pStorage);
3710 if (RT_SUCCESS(rc))
3711 *ppIoStorage = pIoStorage;
3712 else
3713 RTMemFree(pIoStorage);
3714
3715 return rc;
3716}
3717
3718static int vdIOIntCloseLimited(void *pvUser, PVDIOSTORAGE pIoStorage)
3719{
3720 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3721 int rc = pInterfaceIOCallbacks->pfnClose(NULL, pIoStorage->pStorage);
3722 AssertRC(rc);
3723
3724 RTMemFree(pIoStorage);
3725 return VINF_SUCCESS;
3726}
3727
3728static int vdIOIntDeleteLimited(void *pvUser, const char *pcszFilename)
3729{
3730 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3731 return pInterfaceIOCallbacks->pfnDelete(NULL, pcszFilename);
3732}
3733
3734static int vdIOIntMoveLimited(void *pvUser, const char *pcszSrc,
3735 const char *pcszDst, unsigned fMove)
3736{
3737 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3738 return pInterfaceIOCallbacks->pfnMove(NULL, pcszSrc, pcszDst, fMove);
3739}
3740
3741static int vdIOIntGetFreeSpaceLimited(void *pvUser, const char *pcszFilename,
3742 int64_t *pcbFreeSpace)
3743{
3744 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3745 return pInterfaceIOCallbacks->pfnGetFreeSpace(NULL, pcszFilename, pcbFreeSpace);
3746}
3747
3748static int vdIOIntGetModificationTimeLimited(void *pvUser,
3749 const char *pcszFilename,
3750 PRTTIMESPEC pModificationTime)
3751{
3752 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3753 return pInterfaceIOCallbacks->pfnGetModificationTime(NULL, pcszFilename, pModificationTime);
3754}
3755
3756static int vdIOIntGetSizeLimited(void *pvUser, PVDIOSTORAGE pIoStorage,
3757 uint64_t *pcbSize)
3758{
3759 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3760 return pInterfaceIOCallbacks->pfnGetSize(NULL, pIoStorage->pStorage, pcbSize);
3761}
3762
3763static int vdIOIntSetSizeLimited(void *pvUser, PVDIOSTORAGE pIoStorage,
3764 uint64_t cbSize)
3765{
3766 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3767 return pInterfaceIOCallbacks->pfnSetSize(NULL, pIoStorage->pStorage, cbSize);
3768}
3769
3770static int vdIOIntWriteSyncLimited(void *pvUser, PVDIOSTORAGE pIoStorage,
3771 uint64_t uOffset, const void *pvBuf,
3772 size_t cbWrite, size_t *pcbWritten)
3773{
3774 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3775 return pInterfaceIOCallbacks->pfnWriteSync(NULL, pIoStorage->pStorage, uOffset, pvBuf, cbWrite, pcbWritten);
3776}
3777
3778static int vdIOIntReadSyncLimited(void *pvUser, PVDIOSTORAGE pIoStorage,
3779 uint64_t uOffset, void *pvBuf, size_t cbRead,
3780 size_t *pcbRead)
3781{
3782 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3783 return pInterfaceIOCallbacks->pfnReadSync(NULL, pIoStorage->pStorage, uOffset, pvBuf, cbRead, pcbRead);
3784}
3785
3786static int vdIOIntFlushSyncLimited(void *pvUser, PVDIOSTORAGE pIoStorage)
3787{
3788 PVDINTERFACEIO pInterfaceIOCallbacks = (PVDINTERFACEIO)pvUser;
3789 return pInterfaceIOCallbacks->pfnFlushSync(NULL, pIoStorage->pStorage);
3790}
3791
3792/**
3793 * internal: send output to the log (unconditionally).
3794 */
3795int vdLogMessage(void *pvUser, const char *pszFormat, va_list args)
3796{
3797 NOREF(pvUser);
3798 RTLogPrintfV(pszFormat, args);
3799 return VINF_SUCCESS;
3800}
3801
3802DECLINLINE(int) vdMessageWrapper(PVBOXHDD pDisk, const char *pszFormat, ...)
3803{
3804 va_list va;
3805 va_start(va, pszFormat);
3806 int rc = pDisk->pInterfaceErrorCallbacks->pfnMessage(pDisk->pInterfaceError->pvUser,
3807 pszFormat, va);
3808 va_end(va);
3809 return rc;
3810}
3811
3812
3813/**
3814 * internal: adjust PCHS geometry
3815 */
3816static void vdFixupPCHSGeometry(PVDGEOMETRY pPCHS, uint64_t cbSize)
3817{
3818 /* Fix broken PCHS geometry. Can happen for two reasons: either the backend
3819 * mixes up PCHS and LCHS, or the application used to create the source
3820 * image has put garbage in it. Additionally, if the PCHS geometry covers
3821 * more than the image size, set it back to the default. */
3822 if ( pPCHS->cHeads > 16
3823 || pPCHS->cSectors > 63
3824 || pPCHS->cCylinders == 0
3825 || (uint64_t)pPCHS->cHeads * pPCHS->cSectors * pPCHS->cCylinders * 512 > cbSize)
3826 {
3827 Assert(!(RT_MIN(cbSize / 512 / 16 / 63, 16383) - (uint32_t)RT_MIN(cbSize / 512 / 16 / 63, 16383)));
3828 pPCHS->cCylinders = (uint32_t)RT_MIN(cbSize / 512 / 16 / 63, 16383);
3829 pPCHS->cHeads = 16;
3830 pPCHS->cSectors = 63;
3831 }
3832}
3833
3834/**
3835 * internal: adjust PCHS geometry
3836 */
3837static void vdFixupLCHSGeometry(PVDGEOMETRY pLCHS, uint64_t cbSize)
3838{
3839 /* Fix broken LCHS geometry. Can happen for two reasons: either the backend
3840 * mixes up PCHS and LCHS, or the application used to create the source
3841 * image has put garbage in it. The fix in this case is to clear the LCHS
3842 * geometry to trigger autodetection when it is used next. If the geometry
3843 * already says "please autodetect" (cylinders=0) keep it. */
3844 if ( ( pLCHS->cHeads > 255
3845 || pLCHS->cHeads == 0
3846 || pLCHS->cSectors > 63
3847 || pLCHS->cSectors == 0)
3848 && pLCHS->cCylinders != 0)
3849 {
3850 pLCHS->cCylinders = 0;
3851 pLCHS->cHeads = 0;
3852 pLCHS->cSectors = 0;
3853 }
3854 /* Always recompute the number of cylinders stored in the LCHS
3855 * geometry if it isn't set to "autotedetect" at the moment.
3856 * This is very useful if the destination image size is
3857 * larger or smaller than the source image size. Do not modify
3858 * the number of heads and sectors. Windows guests hate it. */
3859 if ( pLCHS->cCylinders != 0
3860 && pLCHS->cHeads != 0 /* paranoia */
3861 && pLCHS->cSectors != 0 /* paranoia */)
3862 {
3863 Assert(!(RT_MIN(cbSize / 512 / pLCHS->cHeads / pLCHS->cSectors, 1024) - (uint32_t)RT_MIN(cbSize / 512 / pLCHS->cHeads / pLCHS->cSectors, 1024)));
3864 pLCHS->cCylinders = (uint32_t)RT_MIN(cbSize / 512 / pLCHS->cHeads / pLCHS->cSectors, 1024);
3865 }
3866}
3867
3868/**
3869 * Initializes HDD backends.
3870 *
3871 * @returns VBox status code.
3872 */
3873VBOXDDU_DECL(int) VDInit(void)
3874{
3875 int rc = vdAddBackends(aStaticBackends, RT_ELEMENTS(aStaticBackends));
3876 if (RT_SUCCESS(rc))
3877 {
3878 rc = vdAddCacheBackends(aStaticCacheBackends, RT_ELEMENTS(aStaticCacheBackends));
3879 if (RT_SUCCESS(rc))
3880 {
3881 rc = vdLoadDynamicBackends();
3882 if (RT_SUCCESS(rc))
3883 rc = vdLoadDynamicCacheBackends();
3884 }
3885 }
3886 LogRel(("VDInit finished\n"));
3887 return rc;
3888}
3889
3890/**
3891 * Destroys loaded HDD backends.
3892 *
3893 * @returns VBox status code.
3894 */
3895VBOXDDU_DECL(int) VDShutdown(void)
3896{
3897 PVBOXHDDBACKEND *pBackends = g_apBackends;
3898 PVDCACHEBACKEND *pCacheBackends = g_apCacheBackends;
3899 unsigned cBackends = g_cBackends;
3900
3901 if (!pBackends)
3902 return VERR_INTERNAL_ERROR;
3903
3904 g_cBackends = 0;
3905 g_apBackends = NULL;
3906
3907#ifndef VBOX_HDD_NO_DYNAMIC_BACKENDS
3908 for (unsigned i = 0; i < cBackends; i++)
3909 if (pBackends[i]->hPlugin != NIL_RTLDRMOD)
3910 RTLdrClose(pBackends[i]->hPlugin);
3911#endif
3912
3913 /* Clear the supported cache backends. */
3914 cBackends = g_cCacheBackends;
3915 g_cCacheBackends = 0;
3916 g_apCacheBackends = NULL;
3917
3918#ifndef VBOX_HDD_NO_DYNAMIC_BACKENDS
3919 for (unsigned i = 0; i < cBackends; i++)
3920 if (pCacheBackends[i]->hPlugin != NIL_RTLDRMOD)
3921 RTLdrClose(pCacheBackends[i]->hPlugin);
3922#endif
3923
3924 if (pCacheBackends)
3925 RTMemFree(pCacheBackends);
3926 RTMemFree(pBackends);
3927 return VINF_SUCCESS;
3928}
3929
3930
3931/**
3932 * Lists all HDD backends and their capabilities in a caller-provided buffer.
3933 *
3934 * @returns VBox status code.
3935 * VERR_BUFFER_OVERFLOW if not enough space is passed.
3936 * @param cEntriesAlloc Number of list entries available.
3937 * @param pEntries Pointer to array for the entries.
3938 * @param pcEntriesUsed Number of entries returned.
3939 */
3940VBOXDDU_DECL(int) VDBackendInfo(unsigned cEntriesAlloc, PVDBACKENDINFO pEntries,
3941 unsigned *pcEntriesUsed)
3942{
3943 int rc = VINF_SUCCESS;
3944 PRTDIR pPluginDir = NULL;
3945 unsigned cEntries = 0;
3946
3947 LogFlowFunc(("cEntriesAlloc=%u pEntries=%#p pcEntriesUsed=%#p\n", cEntriesAlloc, pEntries, pcEntriesUsed));
3948 /* Check arguments. */
3949 AssertMsgReturn(cEntriesAlloc,
3950 ("cEntriesAlloc=%u\n", cEntriesAlloc),
3951 VERR_INVALID_PARAMETER);
3952 AssertMsgReturn(VALID_PTR(pEntries),
3953 ("pEntries=%#p\n", pEntries),
3954 VERR_INVALID_PARAMETER);
3955 AssertMsgReturn(VALID_PTR(pcEntriesUsed),
3956 ("pcEntriesUsed=%#p\n", pcEntriesUsed),
3957 VERR_INVALID_PARAMETER);
3958 if (!g_apBackends)
3959 VDInit();
3960
3961 if (cEntriesAlloc < g_cBackends)
3962 {
3963 *pcEntriesUsed = g_cBackends;
3964 return VERR_BUFFER_OVERFLOW;
3965 }
3966
3967 for (unsigned i = 0; i < g_cBackends; i++)
3968 {
3969 pEntries[i].pszBackend = g_apBackends[i]->pszBackendName;
3970 pEntries[i].uBackendCaps = g_apBackends[i]->uBackendCaps;
3971 pEntries[i].paFileExtensions = g_apBackends[i]->paFileExtensions;
3972 pEntries[i].paConfigInfo = g_apBackends[i]->paConfigInfo;
3973 pEntries[i].pfnComposeLocation = g_apBackends[i]->pfnComposeLocation;
3974 pEntries[i].pfnComposeName = g_apBackends[i]->pfnComposeName;
3975 }
3976
3977 LogFlowFunc(("returns %Rrc *pcEntriesUsed=%u\n", rc, cEntries));
3978 *pcEntriesUsed = g_cBackends;
3979 return rc;
3980}
3981
3982/**
3983 * Lists the capabilities of a backend identified by its name.
3984 *
3985 * @returns VBox status code.
3986 * @param pszBackend The backend name.
3987 * @param pEntries Pointer to an entry.
3988 */
3989VBOXDDU_DECL(int) VDBackendInfoOne(const char *pszBackend, PVDBACKENDINFO pEntry)
3990{
3991 LogFlowFunc(("pszBackend=%#p pEntry=%#p\n", pszBackend, pEntry));
3992 /* Check arguments. */
3993 AssertMsgReturn(VALID_PTR(pszBackend),
3994 ("pszBackend=%#p\n", pszBackend),
3995 VERR_INVALID_PARAMETER);
3996 AssertMsgReturn(VALID_PTR(pEntry),
3997 ("pEntry=%#p\n", pEntry),
3998 VERR_INVALID_PARAMETER);
3999 if (!g_apBackends)
4000 VDInit();
4001
4002 /* Go through loaded backends. */
4003 for (unsigned i = 0; i < g_cBackends; i++)
4004 {
4005 if (!RTStrICmp(pszBackend, g_apBackends[i]->pszBackendName))
4006 {
4007 pEntry->pszBackend = g_apBackends[i]->pszBackendName;
4008 pEntry->uBackendCaps = g_apBackends[i]->uBackendCaps;
4009 pEntry->paFileExtensions = g_apBackends[i]->paFileExtensions;
4010 pEntry->paConfigInfo = g_apBackends[i]->paConfigInfo;
4011 return VINF_SUCCESS;
4012 }
4013 }
4014
4015 return VERR_NOT_FOUND;
4016}
4017
4018/**
4019 * Allocates and initializes an empty HDD container.
4020 * No image files are opened.
4021 *
4022 * @returns VBox status code.
4023 * @param pVDIfsDisk Pointer to the per-disk VD interface list.
4024 * @param enmType Type of the image container.
4025 * @param ppDisk Where to store the reference to HDD container.
4026 */
4027VBOXDDU_DECL(int) VDCreate(PVDINTERFACE pVDIfsDisk, VDTYPE enmType, PVBOXHDD *ppDisk)
4028{
4029 int rc = VINF_SUCCESS;
4030 PVBOXHDD pDisk = NULL;
4031
4032 LogFlowFunc(("pVDIfsDisk=%#p\n", pVDIfsDisk));
4033 do
4034 {
4035 /* Check arguments. */
4036 AssertMsgBreakStmt(VALID_PTR(ppDisk),
4037 ("ppDisk=%#p\n", ppDisk),
4038 rc = VERR_INVALID_PARAMETER);
4039
4040 pDisk = (PVBOXHDD)RTMemAllocZ(sizeof(VBOXHDD));
4041 if (pDisk)
4042 {
4043 pDisk->u32Signature = VBOXHDDDISK_SIGNATURE;
4044 pDisk->enmType = enmType;
4045 pDisk->cImages = 0;
4046 pDisk->pBase = NULL;
4047 pDisk->pLast = NULL;
4048 pDisk->cbSize = 0;
4049 pDisk->PCHSGeometry.cCylinders = 0;
4050 pDisk->PCHSGeometry.cHeads = 0;
4051 pDisk->PCHSGeometry.cSectors = 0;
4052 pDisk->LCHSGeometry.cCylinders = 0;
4053 pDisk->LCHSGeometry.cHeads = 0;
4054 pDisk->LCHSGeometry.cSectors = 0;
4055 pDisk->pVDIfsDisk = pVDIfsDisk;
4056 pDisk->pInterfaceError = NULL;
4057 pDisk->pInterfaceErrorCallbacks = NULL;
4058 pDisk->pInterfaceThreadSync = NULL;
4059 pDisk->pInterfaceThreadSyncCallbacks = NULL;
4060 pDisk->fLocked = false;
4061 pDisk->pIoCtxLockOwner = NULL;
4062 RTListInit(&pDisk->ListWriteLocked);
4063
4064 /* Create the I/O ctx cache */
4065 rc = RTMemCacheCreate(&pDisk->hMemCacheIoCtx, sizeof(VDIOCTX), 0, UINT32_MAX,
4066 NULL, NULL, NULL, 0);
4067 if (RT_FAILURE(rc))
4068 {
4069 RTMemFree(pDisk);
4070 break;
4071 }
4072
4073 /* Create the I/O task cache */
4074 rc = RTMemCacheCreate(&pDisk->hMemCacheIoTask, sizeof(VDIOTASK), 0, UINT32_MAX,
4075 NULL, NULL, NULL, 0);
4076 if (RT_FAILURE(rc))
4077 {
4078 RTMemCacheDestroy(pDisk->hMemCacheIoCtx);
4079 RTMemFree(pDisk);
4080 break;
4081 }
4082
4083 /* Create critical section. */
4084 rc = RTCritSectInit(&pDisk->CritSect);
4085 if (RT_FAILURE(rc))
4086 {
4087 RTMemCacheDestroy(pDisk->hMemCacheIoCtx);
4088 RTMemCacheDestroy(pDisk->hMemCacheIoTask);
4089 RTMemFree(pDisk);
4090 break;
4091 }
4092
4093 pDisk->pInterfaceError = VDInterfaceGet(pVDIfsDisk, VDINTERFACETYPE_ERROR);
4094 if (pDisk->pInterfaceError)
4095 pDisk->pInterfaceErrorCallbacks = VDGetInterfaceError(pDisk->pInterfaceError);
4096
4097 pDisk->pInterfaceThreadSync = VDInterfaceGet(pVDIfsDisk, VDINTERFACETYPE_THREADSYNC);
4098 if (pDisk->pInterfaceThreadSync)
4099 pDisk->pInterfaceThreadSyncCallbacks = VDGetInterfaceThreadSync(pDisk->pInterfaceThreadSync);
4100
4101 /* Create fallback I/O callback table */
4102 pDisk->VDIIOCallbacks.cbSize = sizeof(VDINTERFACEIO);
4103 pDisk->VDIIOCallbacks.enmInterface = VDINTERFACETYPE_IO;
4104 pDisk->VDIIOCallbacks.pfnOpen = vdIOOpenFallback;
4105 pDisk->VDIIOCallbacks.pfnClose = vdIOCloseFallback;
4106 pDisk->VDIIOCallbacks.pfnDelete = vdIODeleteFallback;
4107 pDisk->VDIIOCallbacks.pfnMove = vdIOMoveFallback;
4108 pDisk->VDIIOCallbacks.pfnGetFreeSpace = vdIOGetFreeSpaceFallback;
4109 pDisk->VDIIOCallbacks.pfnGetModificationTime = vdIOGetModificationTimeFallback;
4110 pDisk->VDIIOCallbacks.pfnGetSize = vdIOGetSizeFallback;
4111 pDisk->VDIIOCallbacks.pfnSetSize = vdIOSetSizeFallback;
4112 pDisk->VDIIOCallbacks.pfnReadSync = vdIOReadSyncFallback;
4113 pDisk->VDIIOCallbacks.pfnWriteSync = vdIOWriteSyncFallback;
4114 pDisk->VDIIOCallbacks.pfnFlushSync = vdIOFlushSyncFallback;
4115 pDisk->VDIIOCallbacks.pfnReadAsync = vdIOReadAsyncFallback;
4116 pDisk->VDIIOCallbacks.pfnWriteAsync = vdIOWriteAsyncFallback;
4117 pDisk->VDIIOCallbacks.pfnFlushAsync = vdIOFlushAsyncFallback;
4118
4119 /*
4120 * Create the internal I/O callback table.
4121 * The interface is per-image but no need to duplicate the
4122 * callback table every time.
4123 */
4124 pDisk->VDIIOIntCallbacks.cbSize = sizeof(VDINTERFACEIOINT);
4125 pDisk->VDIIOIntCallbacks.enmInterface = VDINTERFACETYPE_IOINT;
4126 pDisk->VDIIOIntCallbacks.pfnOpen = vdIOIntOpen;
4127 pDisk->VDIIOIntCallbacks.pfnClose = vdIOIntClose;
4128 pDisk->VDIIOIntCallbacks.pfnDelete = vdIOIntDelete;
4129 pDisk->VDIIOIntCallbacks.pfnMove = vdIOIntMove;
4130 pDisk->VDIIOIntCallbacks.pfnGetFreeSpace = vdIOIntGetFreeSpace;
4131 pDisk->VDIIOIntCallbacks.pfnGetModificationTime = vdIOIntGetModificationTime;
4132 pDisk->VDIIOIntCallbacks.pfnGetSize = vdIOIntGetSize;
4133 pDisk->VDIIOIntCallbacks.pfnSetSize = vdIOIntSetSize;
4134 pDisk->VDIIOIntCallbacks.pfnReadSync = vdIOIntReadSync;
4135 pDisk->VDIIOIntCallbacks.pfnWriteSync = vdIOIntWriteSync;
4136 pDisk->VDIIOIntCallbacks.pfnFlushSync = vdIOIntFlushSync;
4137 pDisk->VDIIOIntCallbacks.pfnReadUserAsync = vdIOIntReadUserAsync;
4138 pDisk->VDIIOIntCallbacks.pfnWriteUserAsync = vdIOIntWriteUserAsync;
4139 pDisk->VDIIOIntCallbacks.pfnReadMetaAsync = vdIOIntReadMetaAsync;
4140 pDisk->VDIIOIntCallbacks.pfnWriteMetaAsync = vdIOIntWriteMetaAsync;
4141 pDisk->VDIIOIntCallbacks.pfnMetaXferRelease = vdIOIntMetaXferRelease;
4142 pDisk->VDIIOIntCallbacks.pfnFlushAsync = vdIOIntFlushAsync;
4143 pDisk->VDIIOIntCallbacks.pfnIoCtxCopyFrom = vdIOIntIoCtxCopyFrom;
4144 pDisk->VDIIOIntCallbacks.pfnIoCtxCopyTo = vdIOIntIoCtxCopyTo;
4145 pDisk->VDIIOIntCallbacks.pfnIoCtxSet = vdIOIntIoCtxSet;
4146 pDisk->VDIIOIntCallbacks.pfnIoCtxSegArrayCreate = vdIOIntIoCtxSegArrayCreate;
4147 pDisk->VDIIOIntCallbacks.pfnIoCtxCompleted = vdIOIntIoCtxCompleted;
4148
4149 *ppDisk = pDisk;
4150 }
4151 else
4152 {
4153 rc = VERR_NO_MEMORY;
4154 break;
4155 }
4156 } while (0);
4157
4158 LogFlowFunc(("returns %Rrc (pDisk=%#p)\n", rc, pDisk));
4159 return rc;
4160}
4161
4162/**
4163 * Destroys HDD container.
4164 * If container has opened image files they will be closed.
4165 *
4166 * @param pDisk Pointer to HDD container.
4167 */
4168VBOXDDU_DECL(void) VDDestroy(PVBOXHDD pDisk)
4169{
4170 LogFlowFunc(("pDisk=%#p\n", pDisk));
4171 do
4172 {
4173 /* sanity check */
4174 AssertPtrBreak(pDisk);
4175 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
4176 VDCloseAll(pDisk);
4177 RTCritSectDelete(&pDisk->CritSect);
4178 RTMemCacheDestroy(pDisk->hMemCacheIoCtx);
4179 RTMemCacheDestroy(pDisk->hMemCacheIoTask);
4180 RTMemFree(pDisk);
4181 } while (0);
4182 LogFlowFunc(("returns\n"));
4183}
4184
4185/**
4186 * Try to get the backend name which can use this image.
4187 *
4188 * @returns VBox status code.
4189 * VINF_SUCCESS if a plugin was found.
4190 * ppszFormat contains the string which can be used as backend name.
4191 * VERR_NOT_SUPPORTED if no backend was found.
4192 * @param pVDIfsDisk Pointer to the per-disk VD interface list.
4193 * @param pVDIfsImage Pointer to the per-image VD interface list.
4194 * @param pszFilename Name of the image file for which the backend is queried.
4195 * @param ppszFormat Receives pointer of the UTF-8 string which contains the format name.
4196 * The returned pointer must be freed using RTStrFree().
4197 */
4198VBOXDDU_DECL(int) VDGetFormat(PVDINTERFACE pVDIfsDisk, PVDINTERFACE pVDIfsImage,
4199 const char *pszFilename, char **ppszFormat, VDTYPE *penmType)
4200{
4201 int rc = VERR_NOT_SUPPORTED;
4202 VDINTERFACEIOINT VDIIOIntCallbacks;
4203 VDINTERFACE VDIIOInt;
4204 VDINTERFACEIO VDIIOCallbacksFallback;
4205 PVDINTERFACE pInterfaceIO;
4206 PVDINTERFACEIO pInterfaceIOCallbacks;
4207
4208 LogFlowFunc(("pszFilename=\"%s\"\n", pszFilename));
4209 /* Check arguments. */
4210 AssertMsgReturn(VALID_PTR(pszFilename) && *pszFilename,
4211 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
4212 VERR_INVALID_PARAMETER);
4213 AssertMsgReturn(VALID_PTR(ppszFormat),
4214 ("ppszFormat=%#p\n", ppszFormat),
4215 VERR_INVALID_PARAMETER);
4216 AssertMsgReturn(VALID_PTR(ppszFormat),
4217 ("penmType=%#p\n", penmType),
4218 VERR_INVALID_PARAMETER);
4219
4220 if (!g_apBackends)
4221 VDInit();
4222
4223 pInterfaceIO = VDInterfaceGet(pVDIfsImage, VDINTERFACETYPE_IO);
4224 if (!pInterfaceIO)
4225 {
4226 /*
4227 * Caller doesn't provide an I/O interface, create our own using the
4228 * native file API.
4229 */
4230 VDIIOCallbacksFallback.cbSize = sizeof(VDINTERFACEIO);
4231 VDIIOCallbacksFallback.enmInterface = VDINTERFACETYPE_IO;
4232 VDIIOCallbacksFallback.pfnOpen = vdIOOpenFallback;
4233 VDIIOCallbacksFallback.pfnClose = vdIOCloseFallback;
4234 VDIIOCallbacksFallback.pfnDelete = vdIODeleteFallback;
4235 VDIIOCallbacksFallback.pfnMove = vdIOMoveFallback;
4236 VDIIOCallbacksFallback.pfnGetFreeSpace = vdIOGetFreeSpaceFallback;
4237 VDIIOCallbacksFallback.pfnGetModificationTime = vdIOGetModificationTimeFallback;
4238 VDIIOCallbacksFallback.pfnGetSize = vdIOGetSizeFallback;
4239 VDIIOCallbacksFallback.pfnSetSize = vdIOSetSizeFallback;
4240 VDIIOCallbacksFallback.pfnReadSync = vdIOReadSyncFallback;
4241 VDIIOCallbacksFallback.pfnWriteSync = vdIOWriteSyncFallback;
4242 VDIIOCallbacksFallback.pfnFlushSync = vdIOFlushSyncFallback;
4243 pInterfaceIOCallbacks = &VDIIOCallbacksFallback;
4244 }
4245 else
4246 pInterfaceIOCallbacks = VDGetInterfaceIO(pInterfaceIO);
4247
4248 /* Set up the internal I/O interface. */
4249 AssertReturn(!VDInterfaceGet(pVDIfsImage, VDINTERFACETYPE_IOINT),
4250 VERR_INVALID_PARAMETER);
4251 VDIIOIntCallbacks.cbSize = sizeof(VDINTERFACEIOINT);
4252 VDIIOIntCallbacks.enmInterface = VDINTERFACETYPE_IOINT;
4253 VDIIOIntCallbacks.pfnOpen = vdIOIntOpenLimited;
4254 VDIIOIntCallbacks.pfnClose = vdIOIntCloseLimited;
4255 VDIIOIntCallbacks.pfnDelete = vdIOIntDeleteLimited;
4256 VDIIOIntCallbacks.pfnMove = vdIOIntMoveLimited;
4257 VDIIOIntCallbacks.pfnGetFreeSpace = vdIOIntGetFreeSpaceLimited;
4258 VDIIOIntCallbacks.pfnGetModificationTime = vdIOIntGetModificationTimeLimited;
4259 VDIIOIntCallbacks.pfnGetSize = vdIOIntGetSizeLimited;
4260 VDIIOIntCallbacks.pfnSetSize = vdIOIntSetSizeLimited;
4261 VDIIOIntCallbacks.pfnReadSync = vdIOIntReadSyncLimited;
4262 VDIIOIntCallbacks.pfnWriteSync = vdIOIntWriteSyncLimited;
4263 VDIIOIntCallbacks.pfnFlushSync = vdIOIntFlushSyncLimited;
4264 VDIIOIntCallbacks.pfnReadUserAsync = NULL;
4265 VDIIOIntCallbacks.pfnWriteUserAsync = NULL;
4266 VDIIOIntCallbacks.pfnReadMetaAsync = NULL;
4267 VDIIOIntCallbacks.pfnWriteMetaAsync = NULL;
4268 VDIIOIntCallbacks.pfnFlushAsync = NULL;
4269 rc = VDInterfaceAdd(&VDIIOInt, "VD_IOINT", VDINTERFACETYPE_IOINT,
4270 &VDIIOIntCallbacks, pInterfaceIOCallbacks, &pVDIfsImage);
4271 AssertRC(rc);
4272
4273 /* Find the backend supporting this file format. */
4274 for (unsigned i = 0; i < g_cBackends; i++)
4275 {
4276 if (g_apBackends[i]->pfnCheckIfValid)
4277 {
4278 rc = g_apBackends[i]->pfnCheckIfValid(pszFilename, pVDIfsDisk,
4279 pVDIfsImage, penmType);
4280 if ( RT_SUCCESS(rc)
4281 /* The correct backend has been found, but there is a small
4282 * incompatibility so that the file cannot be used. Stop here
4283 * and signal success - the actual open will of course fail,
4284 * but that will create a really sensible error message. */
4285 || ( rc != VERR_VD_GEN_INVALID_HEADER
4286 && rc != VERR_VD_VDI_INVALID_HEADER
4287 && rc != VERR_VD_VMDK_INVALID_HEADER
4288 && rc != VERR_VD_ISCSI_INVALID_HEADER
4289 && rc != VERR_VD_VHD_INVALID_HEADER
4290 && rc != VERR_VD_RAW_INVALID_HEADER
4291 && rc != VERR_VD_PARALLELS_INVALID_HEADER
4292 && rc != VERR_VD_DMG_INVALID_HEADER))
4293 {
4294 /* Copy the name into the new string. */
4295 char *pszFormat = RTStrDup(g_apBackends[i]->pszBackendName);
4296 if (!pszFormat)
4297 {
4298 rc = VERR_NO_MEMORY;
4299 break;
4300 }
4301 *ppszFormat = pszFormat;
4302 rc = VINF_SUCCESS;
4303 break;
4304 }
4305 rc = VERR_NOT_SUPPORTED;
4306 }
4307 }
4308
4309 /* Try the cache backends. */
4310 if (rc == VERR_NOT_SUPPORTED)
4311 {
4312 for (unsigned i = 0; i < g_cCacheBackends; i++)
4313 {
4314 if (g_apCacheBackends[i]->pfnProbe)
4315 {
4316 rc = g_apCacheBackends[i]->pfnProbe(pszFilename, pVDIfsDisk,
4317 pVDIfsImage);
4318 if ( RT_SUCCESS(rc)
4319 || (rc != VERR_VD_GEN_INVALID_HEADER))
4320 {
4321 /* Copy the name into the new string. */
4322 char *pszFormat = RTStrDup(g_apBackends[i]->pszBackendName);
4323 if (!pszFormat)
4324 {
4325 rc = VERR_NO_MEMORY;
4326 break;
4327 }
4328 *ppszFormat = pszFormat;
4329 rc = VINF_SUCCESS;
4330 break;
4331 }
4332 rc = VERR_NOT_SUPPORTED;
4333 }
4334 }
4335 }
4336
4337 LogFlowFunc(("returns %Rrc *ppszFormat=\"%s\"\n", rc, *ppszFormat));
4338 return rc;
4339}
4340
4341/**
4342 * Opens an image file.
4343 *
4344 * The first opened image file in HDD container must have a base image type,
4345 * others (next opened images) must be a differencing or undo images.
4346 * Linkage is checked for differencing image to be in consistence with the previously opened image.
4347 * When another differencing image is opened and the last image was opened in read/write access
4348 * mode, then the last image is reopened in read-only with deny write sharing mode. This allows
4349 * other processes to use images in read-only mode too.
4350 *
4351 * Note that the image is opened in read-only mode if a read/write open is not possible.
4352 * Use VDIsReadOnly to check open mode.
4353 *
4354 * @returns VBox status code.
4355 * @param pDisk Pointer to HDD container.
4356 * @param pszBackend Name of the image file backend to use.
4357 * @param pszFilename Name of the image file to open.
4358 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
4359 * @param pVDIfsImage Pointer to the per-image VD interface list.
4360 */
4361VBOXDDU_DECL(int) VDOpen(PVBOXHDD pDisk, const char *pszBackend,
4362 const char *pszFilename, unsigned uOpenFlags,
4363 PVDINTERFACE pVDIfsImage)
4364{
4365 int rc = VINF_SUCCESS;
4366 int rc2;
4367 bool fLockWrite = false;
4368 PVDIMAGE pImage = NULL;
4369
4370 LogFlowFunc(("pDisk=%#p pszBackend=\"%s\" pszFilename=\"%s\" uOpenFlags=%#x, pVDIfsImage=%#p\n",
4371 pDisk, pszBackend, pszFilename, uOpenFlags, pVDIfsImage));
4372
4373 do
4374 {
4375 /* sanity check */
4376 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
4377 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
4378
4379 /* Check arguments. */
4380 AssertMsgBreakStmt(VALID_PTR(pszBackend) && *pszBackend,
4381 ("pszBackend=%#p \"%s\"\n", pszBackend, pszBackend),
4382 rc = VERR_INVALID_PARAMETER);
4383 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
4384 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
4385 rc = VERR_INVALID_PARAMETER);
4386 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
4387 ("uOpenFlags=%#x\n", uOpenFlags),
4388 rc = VERR_INVALID_PARAMETER);
4389
4390 /* Set up image descriptor. */
4391 pImage = (PVDIMAGE)RTMemAllocZ(sizeof(VDIMAGE));
4392 if (!pImage)
4393 {
4394 rc = VERR_NO_MEMORY;
4395 break;
4396 }
4397 pImage->pszFilename = RTStrDup(pszFilename);
4398 if (!pImage->pszFilename)
4399 {
4400 rc = VERR_NO_MEMORY;
4401 break;
4402 }
4403
4404 pImage->VDIo.pDisk = pDisk;
4405 pImage->pVDIfsImage = pVDIfsImage;
4406
4407 rc = vdFindBackend(pszBackend, &pImage->Backend);
4408 if (RT_FAILURE(rc))
4409 break;
4410 if (!pImage->Backend)
4411 {
4412 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
4413 N_("VD: unknown backend name '%s'"), pszBackend);
4414 break;
4415 }
4416
4417 /*
4418 * Fail if the the backend can't do async I/O but the
4419 * flag is set.
4420 */
4421 if ( !(pImage->Backend->uBackendCaps & VD_CAP_ASYNC)
4422 && (uOpenFlags & VD_OPEN_FLAGS_ASYNC_IO))
4423 {
4424 rc = vdError(pDisk, VERR_NOT_SUPPORTED, RT_SRC_POS,
4425 N_("VD: Backend '%s' does not support async I/O"), pszBackend);
4426 break;
4427 }
4428
4429 /* Set up the I/O interface. */
4430 pImage->VDIo.pInterfaceIO = VDInterfaceGet(pVDIfsImage, VDINTERFACETYPE_IO);
4431 if (pImage->VDIo.pInterfaceIO)
4432 pImage->VDIo.pInterfaceIOCallbacks = VDGetInterfaceIO(pImage->VDIo.pInterfaceIO);
4433 else
4434 {
4435 rc = VDInterfaceAdd(&pImage->VDIo.VDIIO, "VD_IO", VDINTERFACETYPE_IO,
4436 &pDisk->VDIIOCallbacks, pDisk, &pVDIfsImage);
4437 pImage->VDIo.pInterfaceIO = &pImage->VDIo.VDIIO;
4438 pImage->VDIo.pInterfaceIOCallbacks = &pDisk->VDIIOCallbacks;
4439 }
4440
4441 /* Set up the internal I/O interface. */
4442 AssertBreakStmt(!VDInterfaceGet(pVDIfsImage, VDINTERFACETYPE_IOINT),
4443 rc = VERR_INVALID_PARAMETER);
4444 rc = VDInterfaceAdd(&pImage->VDIo.VDIIOInt, "VD_IOINT", VDINTERFACETYPE_IOINT,
4445 &pDisk->VDIIOIntCallbacks, &pImage->VDIo, &pImage->pVDIfsImage);
4446 AssertRC(rc);
4447
4448 pImage->uOpenFlags = uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME;
4449 rc = pImage->Backend->pfnOpen(pImage->pszFilename,
4450 uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME,
4451 pDisk->pVDIfsDisk,
4452 pImage->pVDIfsImage,
4453 pDisk->enmType,
4454 &pImage->pBackendData);
4455 /* If the open in read-write mode failed, retry in read-only mode. */
4456 if (RT_FAILURE(rc))
4457 {
4458 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY)
4459 && ( rc == VERR_ACCESS_DENIED
4460 || rc == VERR_PERMISSION_DENIED
4461 || rc == VERR_WRITE_PROTECT
4462 || rc == VERR_SHARING_VIOLATION
4463 || rc == VERR_FILE_LOCK_FAILED))
4464 rc = pImage->Backend->pfnOpen(pImage->pszFilename,
4465 (uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME)
4466 | VD_OPEN_FLAGS_READONLY,
4467 pDisk->pVDIfsDisk,
4468 pImage->pVDIfsImage,
4469 pDisk->enmType,
4470 &pImage->pBackendData);
4471 if (RT_FAILURE(rc))
4472 {
4473 rc = vdError(pDisk, rc, RT_SRC_POS,
4474 N_("VD: error %Rrc opening image file '%s'"), rc, pszFilename);
4475 break;
4476 }
4477 }
4478
4479 /* Lock disk for writing, as we modify pDisk information below. */
4480 rc2 = vdThreadStartWrite(pDisk);
4481 AssertRC(rc2);
4482 fLockWrite = true;
4483
4484 pImage->VDIo.pBackendData = pImage->pBackendData;
4485
4486 /* Check image type. As the image itself has only partial knowledge
4487 * whether it's a base image or not, this info is derived here. The
4488 * base image can be fixed or normal, all others must be normal or
4489 * diff images. Some image formats don't distinguish between normal
4490 * and diff images, so this must be corrected here. */
4491 unsigned uImageFlags;
4492 uImageFlags = pImage->Backend->pfnGetImageFlags(pImage->pBackendData);
4493 if (RT_FAILURE(rc))
4494 uImageFlags = VD_IMAGE_FLAGS_NONE;
4495 if ( RT_SUCCESS(rc)
4496 && !(uOpenFlags & VD_OPEN_FLAGS_INFO))
4497 {
4498 if ( pDisk->cImages == 0
4499 && (uImageFlags & VD_IMAGE_FLAGS_DIFF))
4500 {
4501 rc = VERR_VD_INVALID_TYPE;
4502 break;
4503 }
4504 else if (pDisk->cImages != 0)
4505 {
4506 if (uImageFlags & VD_IMAGE_FLAGS_FIXED)
4507 {
4508 rc = VERR_VD_INVALID_TYPE;
4509 break;
4510 }
4511 else
4512 uImageFlags |= VD_IMAGE_FLAGS_DIFF;
4513 }
4514 }
4515
4516 /* Ensure we always get correct diff information, even if the backend
4517 * doesn't actually have a stored flag for this. It must not return
4518 * bogus information for the parent UUID if it is not a diff image. */
4519 RTUUID parentUuid;
4520 RTUuidClear(&parentUuid);
4521 rc2 = pImage->Backend->pfnGetParentUuid(pImage->pBackendData, &parentUuid);
4522 if (RT_SUCCESS(rc2) && !RTUuidIsNull(&parentUuid))
4523 uImageFlags |= VD_IMAGE_FLAGS_DIFF;
4524
4525 pImage->uImageFlags = uImageFlags;
4526
4527 /* Force sane optimization settings. It's not worth avoiding writes
4528 * to fixed size images. The overhead would have almost no payback. */
4529 if (uImageFlags & VD_IMAGE_FLAGS_FIXED)
4530 pImage->uOpenFlags |= VD_OPEN_FLAGS_HONOR_SAME;
4531
4532 /** @todo optionally check UUIDs */
4533
4534 /* Cache disk information. */
4535 pDisk->cbSize = pImage->Backend->pfnGetSize(pImage->pBackendData);
4536
4537 /* Cache PCHS geometry. */
4538 rc2 = pImage->Backend->pfnGetPCHSGeometry(pImage->pBackendData,
4539 &pDisk->PCHSGeometry);
4540 if (RT_FAILURE(rc2))
4541 {
4542 pDisk->PCHSGeometry.cCylinders = 0;
4543 pDisk->PCHSGeometry.cHeads = 0;
4544 pDisk->PCHSGeometry.cSectors = 0;
4545 }
4546 else
4547 {
4548 /* Make sure the PCHS geometry is properly clipped. */
4549 pDisk->PCHSGeometry.cCylinders = RT_MIN(pDisk->PCHSGeometry.cCylinders, 16383);
4550 pDisk->PCHSGeometry.cHeads = RT_MIN(pDisk->PCHSGeometry.cHeads, 16);
4551 pDisk->PCHSGeometry.cSectors = RT_MIN(pDisk->PCHSGeometry.cSectors, 63);
4552 }
4553
4554 /* Cache LCHS geometry. */
4555 rc2 = pImage->Backend->pfnGetLCHSGeometry(pImage->pBackendData,
4556 &pDisk->LCHSGeometry);
4557 if (RT_FAILURE(rc2))
4558 {
4559 pDisk->LCHSGeometry.cCylinders = 0;
4560 pDisk->LCHSGeometry.cHeads = 0;
4561 pDisk->LCHSGeometry.cSectors = 0;
4562 }
4563 else
4564 {
4565 /* Make sure the LCHS geometry is properly clipped. */
4566 pDisk->LCHSGeometry.cHeads = RT_MIN(pDisk->LCHSGeometry.cHeads, 255);
4567 pDisk->LCHSGeometry.cSectors = RT_MIN(pDisk->LCHSGeometry.cSectors, 63);
4568 }
4569
4570 if (pDisk->cImages != 0)
4571 {
4572 /* Switch previous image to read-only mode. */
4573 unsigned uOpenFlagsPrevImg;
4574 uOpenFlagsPrevImg = pDisk->pLast->Backend->pfnGetOpenFlags(pDisk->pLast->pBackendData);
4575 if (!(uOpenFlagsPrevImg & VD_OPEN_FLAGS_READONLY))
4576 {
4577 uOpenFlagsPrevImg |= VD_OPEN_FLAGS_READONLY;
4578 rc = pDisk->pLast->Backend->pfnSetOpenFlags(pDisk->pLast->pBackendData, uOpenFlagsPrevImg);
4579 }
4580 }
4581
4582 if (RT_SUCCESS(rc))
4583 {
4584 /* Image successfully opened, make it the last image. */
4585 vdAddImageToList(pDisk, pImage);
4586 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY))
4587 pDisk->uModified = VD_IMAGE_MODIFIED_FIRST;
4588 }
4589 else
4590 {
4591 /* Error detected, but image opened. Close image. */
4592 rc2 = pImage->Backend->pfnClose(pImage->pBackendData, false);
4593 AssertRC(rc2);
4594 pImage->pBackendData = NULL;
4595 }
4596 } while (0);
4597
4598 if (RT_UNLIKELY(fLockWrite))
4599 {
4600 rc2 = vdThreadFinishWrite(pDisk);
4601 AssertRC(rc2);
4602 }
4603
4604 if (RT_FAILURE(rc))
4605 {
4606 if (pImage)
4607 {
4608 if (pImage->pszFilename)
4609 RTStrFree(pImage->pszFilename);
4610 RTMemFree(pImage);
4611 }
4612 }
4613
4614 LogFlowFunc(("returns %Rrc\n", rc));
4615 return rc;
4616}
4617
4618/**
4619 * Opens a cache image.
4620 *
4621 * @return VBox status code.
4622 * @param pDisk Pointer to the HDD container which should use the cache image.
4623 * @param pszBackend Name of the cache file backend to use (case insensitive).
4624 * @param pszFilename Name of the cache image to open.
4625 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
4626 * @param pVDIfsCache Pointer to the per-cache VD interface list.
4627 */
4628VBOXDDU_DECL(int) VDCacheOpen(PVBOXHDD pDisk, const char *pszBackend,
4629 const char *pszFilename, unsigned uOpenFlags,
4630 PVDINTERFACE pVDIfsCache)
4631{
4632 int rc = VINF_SUCCESS;
4633 int rc2;
4634 bool fLockWrite = false;
4635 PVDCACHE pCache = NULL;
4636
4637 LogFlowFunc(("pDisk=%#p pszBackend=\"%s\" pszFilename=\"%s\" uOpenFlags=%#x, pVDIfsCache=%#p\n",
4638 pDisk, pszBackend, pszFilename, uOpenFlags, pVDIfsCache));
4639
4640 do
4641 {
4642 /* sanity check */
4643 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
4644 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
4645
4646 /* Check arguments. */
4647 AssertMsgBreakStmt(VALID_PTR(pszBackend) && *pszBackend,
4648 ("pszBackend=%#p \"%s\"\n", pszBackend, pszBackend),
4649 rc = VERR_INVALID_PARAMETER);
4650 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
4651 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
4652 rc = VERR_INVALID_PARAMETER);
4653 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
4654 ("uOpenFlags=%#x\n", uOpenFlags),
4655 rc = VERR_INVALID_PARAMETER);
4656
4657 /* Set up image descriptor. */
4658 pCache = (PVDCACHE)RTMemAllocZ(sizeof(VDCACHE));
4659 if (!pCache)
4660 {
4661 rc = VERR_NO_MEMORY;
4662 break;
4663 }
4664 pCache->pszFilename = RTStrDup(pszFilename);
4665 if (!pCache->pszFilename)
4666 {
4667 rc = VERR_NO_MEMORY;
4668 break;
4669 }
4670
4671 pCache->VDIo.pDisk = pDisk;
4672 pCache->pVDIfsCache = pVDIfsCache;
4673
4674 rc = vdFindCacheBackend(pszBackend, &pCache->Backend);
4675 if (RT_FAILURE(rc))
4676 break;
4677 if (!pCache->Backend)
4678 {
4679 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
4680 N_("VD: unknown backend name '%s'"), pszBackend);
4681 break;
4682 }
4683
4684 /* Set up the I/O interface. */
4685 pCache->VDIo.pInterfaceIO = VDInterfaceGet(pVDIfsCache, VDINTERFACETYPE_IO);
4686 if (pCache->VDIo.pInterfaceIO)
4687 pCache->VDIo.pInterfaceIOCallbacks = VDGetInterfaceIO(pCache->VDIo.pInterfaceIO);
4688 else
4689 {
4690 rc = VDInterfaceAdd(&pCache->VDIo.VDIIO, "VD_IO", VDINTERFACETYPE_IO,
4691 &pDisk->VDIIOCallbacks, pDisk, &pVDIfsCache);
4692 pCache->VDIo.pInterfaceIO = &pCache->VDIo.VDIIO;
4693 pCache->VDIo.pInterfaceIOCallbacks = &pDisk->VDIIOCallbacks;
4694 }
4695
4696 /* Set up the internal I/O interface. */
4697 AssertBreakStmt(!VDInterfaceGet(pVDIfsCache, VDINTERFACETYPE_IOINT),
4698 rc = VERR_INVALID_PARAMETER);
4699 rc = VDInterfaceAdd(&pCache->VDIo.VDIIOInt, "VD_IOINT", VDINTERFACETYPE_IOINT,
4700 &pDisk->VDIIOIntCallbacks, &pCache->VDIo, &pCache->pVDIfsCache);
4701 AssertRC(rc);
4702
4703 pCache->uOpenFlags = uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME;
4704 rc = pCache->Backend->pfnOpen(pCache->pszFilename,
4705 uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME,
4706 pDisk->pVDIfsDisk,
4707 pCache->pVDIfsCache,
4708 &pCache->pBackendData);
4709 /* If the open in read-write mode failed, retry in read-only mode. */
4710 if (RT_FAILURE(rc))
4711 {
4712 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY)
4713 && ( rc == VERR_ACCESS_DENIED
4714 || rc == VERR_PERMISSION_DENIED
4715 || rc == VERR_WRITE_PROTECT
4716 || rc == VERR_SHARING_VIOLATION
4717 || rc == VERR_FILE_LOCK_FAILED))
4718 rc = pCache->Backend->pfnOpen(pCache->pszFilename,
4719 (uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME)
4720 | VD_OPEN_FLAGS_READONLY,
4721 pDisk->pVDIfsDisk,
4722 pCache->pVDIfsCache,
4723 &pCache->pBackendData);
4724 if (RT_FAILURE(rc))
4725 {
4726 rc = vdError(pDisk, rc, RT_SRC_POS,
4727 N_("VD: error %Rrc opening image file '%s'"), rc, pszFilename);
4728 break;
4729 }
4730 }
4731
4732 /* Lock disk for writing, as we modify pDisk information below. */
4733 rc2 = vdThreadStartWrite(pDisk);
4734 AssertRC(rc2);
4735 fLockWrite = true;
4736
4737 /*
4738 * Check that the modification UUID of the cache and last image
4739 * match. If not the image was modified in-between without the cache.
4740 * The cache might contain stale data.
4741 */
4742 RTUUID UuidImage, UuidCache;
4743
4744 rc = pCache->Backend->pfnGetModificationUuid(pCache->pBackendData,
4745 &UuidCache);
4746 if (RT_SUCCESS(rc))
4747 {
4748 rc = pDisk->pLast->Backend->pfnGetModificationUuid(pDisk->pLast->pBackendData,
4749 &UuidImage);
4750 if (RT_SUCCESS(rc))
4751 {
4752 if (RTUuidCompare(&UuidImage, &UuidCache))
4753 rc = VERR_VD_CACHE_NOT_UP_TO_DATE;
4754 }
4755 }
4756
4757 /*
4758 * We assume that the user knows what he is doing if one of the images
4759 * doesn't support the modification uuid.
4760 */
4761 if (rc == VERR_NOT_SUPPORTED)
4762 rc = VINF_SUCCESS;
4763
4764 if (RT_SUCCESS(rc))
4765 {
4766 /* Cache successfully opened, make it the current one. */
4767 if (!pDisk->pCache)
4768 pDisk->pCache = pCache;
4769 else
4770 rc = VERR_VD_CACHE_ALREADY_EXISTS;
4771 }
4772
4773 if (RT_FAILURE(rc))
4774 {
4775 /* Error detected, but image opened. Close image. */
4776 rc2 = pCache->Backend->pfnClose(pCache->pBackendData, false);
4777 AssertRC(rc2);
4778 pCache->pBackendData = NULL;
4779 }
4780 } while (0);
4781
4782 if (RT_UNLIKELY(fLockWrite))
4783 {
4784 rc2 = vdThreadFinishWrite(pDisk);
4785 AssertRC(rc2);
4786 }
4787
4788 if (RT_FAILURE(rc))
4789 {
4790 if (pCache)
4791 {
4792 if (pCache->pszFilename)
4793 RTStrFree(pCache->pszFilename);
4794 RTMemFree(pCache);
4795 }
4796 }
4797
4798 LogFlowFunc(("returns %Rrc\n", rc));
4799 return rc;
4800}
4801
4802/**
4803 * Creates and opens a new base image file.
4804 *
4805 * @returns VBox status code.
4806 * @param pDisk Pointer to HDD container.
4807 * @param pszBackend Name of the image file backend to use.
4808 * @param pszFilename Name of the image file to create.
4809 * @param cbSize Image size in bytes.
4810 * @param uImageFlags Flags specifying special image features.
4811 * @param pszComment Pointer to image comment. NULL is ok.
4812 * @param pPCHSGeometry Pointer to physical disk geometry <= (16383,16,63). Not NULL.
4813 * @param pLCHSGeometry Pointer to logical disk geometry <= (x,255,63). Not NULL.
4814 * @param pUuid New UUID of the image. If NULL, a new UUID is created.
4815 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
4816 * @param pVDIfsImage Pointer to the per-image VD interface list.
4817 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
4818 */
4819VBOXDDU_DECL(int) VDCreateBase(PVBOXHDD pDisk, const char *pszBackend,
4820 const char *pszFilename, uint64_t cbSize,
4821 unsigned uImageFlags, const char *pszComment,
4822 PCVDGEOMETRY pPCHSGeometry,
4823 PCVDGEOMETRY pLCHSGeometry,
4824 PCRTUUID pUuid, unsigned uOpenFlags,
4825 PVDINTERFACE pVDIfsImage,
4826 PVDINTERFACE pVDIfsOperation)
4827{
4828 int rc = VINF_SUCCESS;
4829 int rc2;
4830 bool fLockWrite = false, fLockRead = false;
4831 PVDIMAGE pImage = NULL;
4832 RTUUID uuid;
4833
4834 LogFlowFunc(("pDisk=%#p pszBackend=\"%s\" pszFilename=\"%s\" cbSize=%llu uImageFlags=%#x pszComment=\"%s\" PCHS=%u/%u/%u LCHS=%u/%u/%u Uuid=%RTuuid uOpenFlags=%#x pVDIfsImage=%#p pVDIfsOperation=%#p\n",
4835 pDisk, pszBackend, pszFilename, cbSize, uImageFlags, pszComment,
4836 pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads,
4837 pPCHSGeometry->cSectors, pLCHSGeometry->cCylinders,
4838 pLCHSGeometry->cHeads, pLCHSGeometry->cSectors, pUuid,
4839 uOpenFlags, pVDIfsImage, pVDIfsOperation));
4840
4841 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
4842 VDINTERFACETYPE_PROGRESS);
4843 PVDINTERFACEPROGRESS pCbProgress = NULL;
4844 if (pIfProgress)
4845 pCbProgress = VDGetInterfaceProgress(pIfProgress);
4846
4847 do
4848 {
4849 /* sanity check */
4850 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
4851 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
4852
4853 /* Check arguments. */
4854 AssertMsgBreakStmt(VALID_PTR(pszBackend) && *pszBackend,
4855 ("pszBackend=%#p \"%s\"\n", pszBackend, pszBackend),
4856 rc = VERR_INVALID_PARAMETER);
4857 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
4858 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
4859 rc = VERR_INVALID_PARAMETER);
4860 AssertMsgBreakStmt(cbSize,
4861 ("cbSize=%llu\n", cbSize),
4862 rc = VERR_INVALID_PARAMETER);
4863 AssertMsgBreakStmt( ((uImageFlags & ~VD_IMAGE_FLAGS_MASK) == 0)
4864 || ((uImageFlags & (VD_IMAGE_FLAGS_FIXED | VD_IMAGE_FLAGS_DIFF)) != VD_IMAGE_FLAGS_FIXED),
4865 ("uImageFlags=%#x\n", uImageFlags),
4866 rc = VERR_INVALID_PARAMETER);
4867 /* The PCHS geometry fields may be 0 to leave it for later. */
4868 AssertMsgBreakStmt( VALID_PTR(pPCHSGeometry)
4869 && pPCHSGeometry->cHeads <= 16
4870 && pPCHSGeometry->cSectors <= 63,
4871 ("pPCHSGeometry=%#p PCHS=%u/%u/%u\n", pPCHSGeometry,
4872 pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads,
4873 pPCHSGeometry->cSectors),
4874 rc = VERR_INVALID_PARAMETER);
4875 /* The LCHS geometry fields may be 0 to leave it to later autodetection. */
4876 AssertMsgBreakStmt( VALID_PTR(pLCHSGeometry)
4877 && pLCHSGeometry->cHeads <= 255
4878 && pLCHSGeometry->cSectors <= 63,
4879 ("pLCHSGeometry=%#p LCHS=%u/%u/%u\n", pLCHSGeometry,
4880 pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads,
4881 pLCHSGeometry->cSectors),
4882 rc = VERR_INVALID_PARAMETER);
4883 /* The UUID may be NULL. */
4884 AssertMsgBreakStmt(pUuid == NULL || VALID_PTR(pUuid),
4885 ("pUuid=%#p UUID=%RTuuid\n", pUuid, pUuid),
4886 rc = VERR_INVALID_PARAMETER);
4887 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
4888 ("uOpenFlags=%#x\n", uOpenFlags),
4889 rc = VERR_INVALID_PARAMETER);
4890
4891 /* Check state. Needs a temporary read lock. Holding the write lock
4892 * all the time would be blocking other activities for too long. */
4893 rc2 = vdThreadStartRead(pDisk);
4894 AssertRC(rc2);
4895 fLockRead = true;
4896 AssertMsgBreakStmt(pDisk->cImages == 0,
4897 ("Create base image cannot be done with other images open\n"),
4898 rc = VERR_VD_INVALID_STATE);
4899 rc2 = vdThreadFinishRead(pDisk);
4900 AssertRC(rc2);
4901 fLockRead = false;
4902
4903 /* Set up image descriptor. */
4904 pImage = (PVDIMAGE)RTMemAllocZ(sizeof(VDIMAGE));
4905 if (!pImage)
4906 {
4907 rc = VERR_NO_MEMORY;
4908 break;
4909 }
4910 pImage->pszFilename = RTStrDup(pszFilename);
4911 if (!pImage->pszFilename)
4912 {
4913 rc = VERR_NO_MEMORY;
4914 break;
4915 }
4916 pImage->VDIo.pDisk = pDisk;
4917 pImage->pVDIfsImage = pVDIfsImage;
4918
4919 /* Set up the I/O interface. */
4920 pImage->VDIo.pInterfaceIO = VDInterfaceGet(pVDIfsImage, VDINTERFACETYPE_IO);
4921 if (pImage->VDIo.pInterfaceIO)
4922 pImage->VDIo.pInterfaceIOCallbacks = VDGetInterfaceIO(pImage->VDIo.pInterfaceIO);
4923 else
4924 {
4925 rc = VDInterfaceAdd(&pImage->VDIo.VDIIO, "VD_IO", VDINTERFACETYPE_IO,
4926 &pDisk->VDIIOCallbacks, pDisk, &pVDIfsImage);
4927 pImage->VDIo.pInterfaceIO = &pImage->VDIo.VDIIO;
4928 pImage->VDIo.pInterfaceIOCallbacks = &pDisk->VDIIOCallbacks;
4929 }
4930
4931 /* Set up the internal I/O interface. */
4932 AssertBreakStmt(!VDInterfaceGet(pVDIfsImage, VDINTERFACETYPE_IOINT),
4933 rc = VERR_INVALID_PARAMETER);
4934 rc = VDInterfaceAdd(&pImage->VDIo.VDIIOInt, "VD_IOINT", VDINTERFACETYPE_IOINT,
4935 &pDisk->VDIIOIntCallbacks, &pImage->VDIo, &pImage->pVDIfsImage);
4936 AssertRC(rc);
4937
4938 rc = vdFindBackend(pszBackend, &pImage->Backend);
4939 if (RT_FAILURE(rc))
4940 break;
4941 if (!pImage->Backend)
4942 {
4943 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
4944 N_("VD: unknown backend name '%s'"), pszBackend);
4945 break;
4946 }
4947 if (!(pImage->Backend->uBackendCaps & ( VD_CAP_CREATE_FIXED
4948 | VD_CAP_CREATE_DYNAMIC)))
4949 {
4950 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
4951 N_("VD: backend '%s' cannot create base images"), pszBackend);
4952 break;
4953 }
4954
4955 /* Create UUID if the caller didn't specify one. */
4956 if (!pUuid)
4957 {
4958 rc = RTUuidCreate(&uuid);
4959 if (RT_FAILURE(rc))
4960 {
4961 rc = vdError(pDisk, rc, RT_SRC_POS,
4962 N_("VD: cannot generate UUID for image '%s'"),
4963 pszFilename);
4964 break;
4965 }
4966 pUuid = &uuid;
4967 }
4968
4969 pImage->uOpenFlags = uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME;
4970 uImageFlags &= ~VD_IMAGE_FLAGS_DIFF;
4971 rc = pImage->Backend->pfnCreate(pImage->pszFilename, cbSize,
4972 uImageFlags, pszComment, pPCHSGeometry,
4973 pLCHSGeometry, pUuid,
4974 uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME,
4975 0, 99,
4976 pDisk->pVDIfsDisk,
4977 pImage->pVDIfsImage,
4978 pVDIfsOperation,
4979 &pImage->pBackendData);
4980
4981 if (RT_SUCCESS(rc))
4982 {
4983 pImage->VDIo.pBackendData = pImage->pBackendData;
4984 pImage->uImageFlags = uImageFlags;
4985
4986 /* Force sane optimization settings. It's not worth avoiding writes
4987 * to fixed size images. The overhead would have almost no payback. */
4988 if (uImageFlags & VD_IMAGE_FLAGS_FIXED)
4989 pImage->uOpenFlags |= VD_OPEN_FLAGS_HONOR_SAME;
4990
4991 /* Lock disk for writing, as we modify pDisk information below. */
4992 rc2 = vdThreadStartWrite(pDisk);
4993 AssertRC(rc2);
4994 fLockWrite = true;
4995
4996 /** @todo optionally check UUIDs */
4997
4998 /* Re-check state, as the lock wasn't held and another image
4999 * creation call could have been done by another thread. */
5000 AssertMsgStmt(pDisk->cImages == 0,
5001 ("Create base image cannot be done with other images open\n"),
5002 rc = VERR_VD_INVALID_STATE);
5003 }
5004
5005 if (RT_SUCCESS(rc))
5006 {
5007 /* Cache disk information. */
5008 pDisk->cbSize = pImage->Backend->pfnGetSize(pImage->pBackendData);
5009
5010 /* Cache PCHS geometry. */
5011 rc2 = pImage->Backend->pfnGetPCHSGeometry(pImage->pBackendData,
5012 &pDisk->PCHSGeometry);
5013 if (RT_FAILURE(rc2))
5014 {
5015 pDisk->PCHSGeometry.cCylinders = 0;
5016 pDisk->PCHSGeometry.cHeads = 0;
5017 pDisk->PCHSGeometry.cSectors = 0;
5018 }
5019 else
5020 {
5021 /* Make sure the CHS geometry is properly clipped. */
5022 pDisk->PCHSGeometry.cCylinders = RT_MIN(pDisk->PCHSGeometry.cCylinders, 16383);
5023 pDisk->PCHSGeometry.cHeads = RT_MIN(pDisk->PCHSGeometry.cHeads, 16);
5024 pDisk->PCHSGeometry.cSectors = RT_MIN(pDisk->PCHSGeometry.cSectors, 63);
5025 }
5026
5027 /* Cache LCHS geometry. */
5028 rc2 = pImage->Backend->pfnGetLCHSGeometry(pImage->pBackendData,
5029 &pDisk->LCHSGeometry);
5030 if (RT_FAILURE(rc2))
5031 {
5032 pDisk->LCHSGeometry.cCylinders = 0;
5033 pDisk->LCHSGeometry.cHeads = 0;
5034 pDisk->LCHSGeometry.cSectors = 0;
5035 }
5036 else
5037 {
5038 /* Make sure the CHS geometry is properly clipped. */
5039 pDisk->LCHSGeometry.cHeads = RT_MIN(pDisk->LCHSGeometry.cHeads, 255);
5040 pDisk->LCHSGeometry.cSectors = RT_MIN(pDisk->LCHSGeometry.cSectors, 63);
5041 }
5042
5043 /* Image successfully opened, make it the last image. */
5044 vdAddImageToList(pDisk, pImage);
5045 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY))
5046 pDisk->uModified = VD_IMAGE_MODIFIED_FIRST;
5047 }
5048 else
5049 {
5050 /* Error detected, image may or may not be opened. Close and delete
5051 * image if it was opened. */
5052 if (pImage->pBackendData)
5053 {
5054 rc2 = pImage->Backend->pfnClose(pImage->pBackendData, true);
5055 AssertRC(rc2);
5056 pImage->pBackendData = NULL;
5057 }
5058 }
5059 } while (0);
5060
5061 if (RT_UNLIKELY(fLockWrite))
5062 {
5063 rc2 = vdThreadFinishWrite(pDisk);
5064 AssertRC(rc2);
5065 }
5066 else if (RT_UNLIKELY(fLockRead))
5067 {
5068 rc2 = vdThreadFinishRead(pDisk);
5069 AssertRC(rc2);
5070 }
5071
5072 if (RT_FAILURE(rc))
5073 {
5074 if (pImage)
5075 {
5076 if (pImage->pszFilename)
5077 RTStrFree(pImage->pszFilename);
5078 RTMemFree(pImage);
5079 }
5080 }
5081
5082 if (RT_SUCCESS(rc) && pCbProgress && pCbProgress->pfnProgress)
5083 pCbProgress->pfnProgress(pIfProgress->pvUser, 100);
5084
5085 LogFlowFunc(("returns %Rrc\n", rc));
5086 return rc;
5087}
5088
5089/**
5090 * Creates and opens a new differencing image file in HDD container.
5091 * See comments for VDOpen function about differencing images.
5092 *
5093 * @returns VBox status code.
5094 * @param pDisk Pointer to HDD container.
5095 * @param pszBackend Name of the image file backend to use.
5096 * @param pszFilename Name of the differencing image file to create.
5097 * @param uImageFlags Flags specifying special image features.
5098 * @param pszComment Pointer to image comment. NULL is ok.
5099 * @param pUuid New UUID of the image. If NULL, a new UUID is created.
5100 * @param pParentUuid New parent UUID of the image. If NULL, the UUID is queried automatically.
5101 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
5102 * @param pVDIfsImage Pointer to the per-image VD interface list.
5103 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
5104 */
5105VBOXDDU_DECL(int) VDCreateDiff(PVBOXHDD pDisk, const char *pszBackend,
5106 const char *pszFilename, unsigned uImageFlags,
5107 const char *pszComment, PCRTUUID pUuid,
5108 PCRTUUID pParentUuid, unsigned uOpenFlags,
5109 PVDINTERFACE pVDIfsImage,
5110 PVDINTERFACE pVDIfsOperation)
5111{
5112 int rc = VINF_SUCCESS;
5113 int rc2;
5114 bool fLockWrite = false, fLockRead = false;
5115 PVDIMAGE pImage = NULL;
5116 RTUUID uuid;
5117
5118 LogFlowFunc(("pDisk=%#p pszBackend=\"%s\" pszFilename=\"%s\" uImageFlags=%#x pszComment=\"%s\" Uuid=%RTuuid uOpenFlags=%#x pVDIfsImage=%#p pVDIfsOperation=%#p\n",
5119 pDisk, pszBackend, pszFilename, uImageFlags, pszComment, pUuid, uOpenFlags, pVDIfsImage, pVDIfsOperation));
5120
5121 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
5122 VDINTERFACETYPE_PROGRESS);
5123 PVDINTERFACEPROGRESS pCbProgress = NULL;
5124 if (pIfProgress)
5125 pCbProgress = VDGetInterfaceProgress(pIfProgress);
5126
5127 do
5128 {
5129 /* sanity check */
5130 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
5131 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
5132
5133 /* Check arguments. */
5134 AssertMsgBreakStmt(VALID_PTR(pszBackend) && *pszBackend,
5135 ("pszBackend=%#p \"%s\"\n", pszBackend, pszBackend),
5136 rc = VERR_INVALID_PARAMETER);
5137 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
5138 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
5139 rc = VERR_INVALID_PARAMETER);
5140 AssertMsgBreakStmt((uImageFlags & ~VD_IMAGE_FLAGS_MASK) == 0,
5141 ("uImageFlags=%#x\n", uImageFlags),
5142 rc = VERR_INVALID_PARAMETER);
5143 /* The UUID may be NULL. */
5144 AssertMsgBreakStmt(pUuid == NULL || VALID_PTR(pUuid),
5145 ("pUuid=%#p UUID=%RTuuid\n", pUuid, pUuid),
5146 rc = VERR_INVALID_PARAMETER);
5147 /* The parent UUID may be NULL. */
5148 AssertMsgBreakStmt(pParentUuid == NULL || VALID_PTR(pParentUuid),
5149 ("pParentUuid=%#p ParentUUID=%RTuuid\n", pParentUuid, pParentUuid),
5150 rc = VERR_INVALID_PARAMETER);
5151 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
5152 ("uOpenFlags=%#x\n", uOpenFlags),
5153 rc = VERR_INVALID_PARAMETER);
5154
5155 /* Check state. Needs a temporary read lock. Holding the write lock
5156 * all the time would be blocking other activities for too long. */
5157 rc2 = vdThreadStartRead(pDisk);
5158 AssertRC(rc2);
5159 fLockRead = true;
5160 AssertMsgBreakStmt(pDisk->cImages != 0,
5161 ("Create diff image cannot be done without other images open\n"),
5162 rc = VERR_VD_INVALID_STATE);
5163 rc2 = vdThreadFinishRead(pDisk);
5164 AssertRC(rc2);
5165 fLockRead = false;
5166
5167 /* Set up image descriptor. */
5168 pImage = (PVDIMAGE)RTMemAllocZ(sizeof(VDIMAGE));
5169 if (!pImage)
5170 {
5171 rc = VERR_NO_MEMORY;
5172 break;
5173 }
5174 pImage->pszFilename = RTStrDup(pszFilename);
5175 if (!pImage->pszFilename)
5176 {
5177 rc = VERR_NO_MEMORY;
5178 break;
5179 }
5180
5181 rc = vdFindBackend(pszBackend, &pImage->Backend);
5182 if (RT_FAILURE(rc))
5183 break;
5184 if (!pImage->Backend)
5185 {
5186 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
5187 N_("VD: unknown backend name '%s'"), pszBackend);
5188 break;
5189 }
5190 if ( !(pImage->Backend->uBackendCaps & VD_CAP_DIFF)
5191 || !(pImage->Backend->uBackendCaps & ( VD_CAP_CREATE_FIXED
5192 | VD_CAP_CREATE_DYNAMIC)))
5193 {
5194 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
5195 N_("VD: backend '%s' cannot create diff images"), pszBackend);
5196 break;
5197 }
5198
5199 pImage->VDIo.pDisk = pDisk;
5200 pImage->pVDIfsImage = pVDIfsImage;
5201
5202 /* Set up the I/O interface. */
5203 pImage->VDIo.pInterfaceIO = VDInterfaceGet(pVDIfsImage, VDINTERFACETYPE_IO);
5204 if (pImage->VDIo.pInterfaceIO)
5205 pImage->VDIo.pInterfaceIOCallbacks = VDGetInterfaceIO(pImage->VDIo.pInterfaceIO);
5206 else
5207 {
5208 rc = VDInterfaceAdd(&pImage->VDIo.VDIIO, "VD_IO", VDINTERFACETYPE_IO,
5209 &pDisk->VDIIOCallbacks, pDisk, &pVDIfsImage);
5210 pImage->VDIo.pInterfaceIO = &pImage->VDIo.VDIIO;
5211 pImage->VDIo.pInterfaceIOCallbacks = &pDisk->VDIIOCallbacks;
5212 }
5213
5214 /* Set up the internal I/O interface. */
5215 AssertBreakStmt(!VDInterfaceGet(pVDIfsImage, VDINTERFACETYPE_IOINT),
5216 rc = VERR_INVALID_PARAMETER);
5217 rc = VDInterfaceAdd(&pImage->VDIo.VDIIOInt, "VD_IOINT", VDINTERFACETYPE_IOINT,
5218 &pDisk->VDIIOIntCallbacks, &pImage->VDIo, &pImage->pVDIfsImage);
5219 AssertRC(rc);
5220
5221 /* Create UUID if the caller didn't specify one. */
5222 if (!pUuid)
5223 {
5224 rc = RTUuidCreate(&uuid);
5225 if (RT_FAILURE(rc))
5226 {
5227 rc = vdError(pDisk, rc, RT_SRC_POS,
5228 N_("VD: cannot generate UUID for image '%s'"),
5229 pszFilename);
5230 break;
5231 }
5232 pUuid = &uuid;
5233 }
5234
5235 pImage->uOpenFlags = uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME;
5236 uImageFlags |= VD_IMAGE_FLAGS_DIFF;
5237 rc = pImage->Backend->pfnCreate(pImage->pszFilename, pDisk->cbSize,
5238 uImageFlags | VD_IMAGE_FLAGS_DIFF,
5239 pszComment, &pDisk->PCHSGeometry,
5240 &pDisk->LCHSGeometry, pUuid,
5241 uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME,
5242 0, 99,
5243 pDisk->pVDIfsDisk,
5244 pImage->pVDIfsImage,
5245 pVDIfsOperation,
5246 &pImage->pBackendData);
5247
5248 if (RT_SUCCESS(rc))
5249 {
5250 pImage->VDIo.pBackendData = pImage->pBackendData;
5251 pImage->uImageFlags = uImageFlags;
5252
5253 /* Lock disk for writing, as we modify pDisk information below. */
5254 rc2 = vdThreadStartWrite(pDisk);
5255 AssertRC(rc2);
5256 fLockWrite = true;
5257
5258 /* Switch previous image to read-only mode. */
5259 unsigned uOpenFlagsPrevImg;
5260 uOpenFlagsPrevImg = pDisk->pLast->Backend->pfnGetOpenFlags(pDisk->pLast->pBackendData);
5261 if (!(uOpenFlagsPrevImg & VD_OPEN_FLAGS_READONLY))
5262 {
5263 uOpenFlagsPrevImg |= VD_OPEN_FLAGS_READONLY;
5264 rc = pDisk->pLast->Backend->pfnSetOpenFlags(pDisk->pLast->pBackendData, uOpenFlagsPrevImg);
5265 }
5266
5267 /** @todo optionally check UUIDs */
5268
5269 /* Re-check state, as the lock wasn't held and another image
5270 * creation call could have been done by another thread. */
5271 AssertMsgStmt(pDisk->cImages != 0,
5272 ("Create diff image cannot be done without other images open\n"),
5273 rc = VERR_VD_INVALID_STATE);
5274 }
5275
5276 if (RT_SUCCESS(rc))
5277 {
5278 RTUUID Uuid;
5279 RTTIMESPEC ts;
5280
5281 if (pParentUuid && !RTUuidIsNull(pParentUuid))
5282 {
5283 Uuid = *pParentUuid;
5284 pImage->Backend->pfnSetParentUuid(pImage->pBackendData, &Uuid);
5285 }
5286 else
5287 {
5288 rc2 = pDisk->pLast->Backend->pfnGetUuid(pDisk->pLast->pBackendData,
5289 &Uuid);
5290 if (RT_SUCCESS(rc2))
5291 pImage->Backend->pfnSetParentUuid(pImage->pBackendData, &Uuid);
5292 }
5293 rc2 = pDisk->pLast->Backend->pfnGetModificationUuid(pDisk->pLast->pBackendData,
5294 &Uuid);
5295 if (RT_SUCCESS(rc2))
5296 pImage->Backend->pfnSetParentModificationUuid(pImage->pBackendData,
5297 &Uuid);
5298 if (pDisk->pLast->Backend->pfnGetTimeStamp)
5299 rc2 = pDisk->pLast->Backend->pfnGetTimeStamp(pDisk->pLast->pBackendData,
5300 &ts);
5301 else
5302 rc2 = VERR_NOT_IMPLEMENTED;
5303 if (RT_SUCCESS(rc2) && pImage->Backend->pfnSetParentTimeStamp)
5304 pImage->Backend->pfnSetParentTimeStamp(pImage->pBackendData, &ts);
5305
5306 if (pImage->Backend->pfnSetParentFilename)
5307 rc2 = pImage->Backend->pfnSetParentFilename(pImage->pBackendData, pDisk->pLast->pszFilename);
5308 }
5309
5310 if (RT_SUCCESS(rc))
5311 {
5312 /* Image successfully opened, make it the last image. */
5313 vdAddImageToList(pDisk, pImage);
5314 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY))
5315 pDisk->uModified = VD_IMAGE_MODIFIED_FIRST;
5316 }
5317 else
5318 {
5319 /* Error detected, but image opened. Close and delete image. */
5320 rc2 = pImage->Backend->pfnClose(pImage->pBackendData, true);
5321 AssertRC(rc2);
5322 pImage->pBackendData = NULL;
5323 }
5324 } while (0);
5325
5326 if (RT_UNLIKELY(fLockWrite))
5327 {
5328 rc2 = vdThreadFinishWrite(pDisk);
5329 AssertRC(rc2);
5330 }
5331 else if (RT_UNLIKELY(fLockRead))
5332 {
5333 rc2 = vdThreadFinishRead(pDisk);
5334 AssertRC(rc2);
5335 }
5336
5337 if (RT_FAILURE(rc))
5338 {
5339 if (pImage)
5340 {
5341 if (pImage->pszFilename)
5342 RTStrFree(pImage->pszFilename);
5343 RTMemFree(pImage);
5344 }
5345 }
5346
5347 if (RT_SUCCESS(rc) && pCbProgress && pCbProgress->pfnProgress)
5348 pCbProgress->pfnProgress(pIfProgress->pvUser, 100);
5349
5350 LogFlowFunc(("returns %Rrc\n", rc));
5351 return rc;
5352}
5353
5354
5355/**
5356 * Creates and opens new cache image file in HDD container.
5357 *
5358 * @return VBox status code.
5359 * @param pDisk Name of the cache file backend to use (case insensitive).
5360 * @param pszFilename Name of the differencing cache file to create.
5361 * @param cbSize Maximum size of the cache.
5362 * @param uImageFlags Flags specifying special cache features.
5363 * @param pszComment Pointer to image comment. NULL is ok.
5364 * @param pUuid New UUID of the image. If NULL, a new UUID is created.
5365 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
5366 * @param pVDIfsCache Pointer to the per-cache VD interface list.
5367 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
5368 */
5369VBOXDDU_DECL(int) VDCreateCache(PVBOXHDD pDisk, const char *pszBackend,
5370 const char *pszFilename, uint64_t cbSize,
5371 unsigned uImageFlags, const char *pszComment,
5372 PCRTUUID pUuid, unsigned uOpenFlags,
5373 PVDINTERFACE pVDIfsCache, PVDINTERFACE pVDIfsOperation)
5374{
5375 int rc = VINF_SUCCESS;
5376 int rc2;
5377 bool fLockWrite = false, fLockRead = false;
5378 PVDCACHE pCache = NULL;
5379 RTUUID uuid;
5380
5381 LogFlowFunc(("pDisk=%#p pszBackend=\"%s\" pszFilename=\"%s\" cbSize=%llu uImageFlags=%#x pszComment=\"%s\" Uuid=%RTuuid uOpenFlags=%#x pVDIfsImage=%#p pVDIfsOperation=%#p\n",
5382 pDisk, pszBackend, pszFilename, cbSize, uImageFlags, pszComment, pUuid, uOpenFlags, pVDIfsCache, pVDIfsOperation));
5383
5384 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
5385 VDINTERFACETYPE_PROGRESS);
5386 PVDINTERFACEPROGRESS pCbProgress = NULL;
5387 if (pIfProgress)
5388 pCbProgress = VDGetInterfaceProgress(pIfProgress);
5389
5390 do
5391 {
5392 /* sanity check */
5393 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
5394 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
5395
5396 /* Check arguments. */
5397 AssertMsgBreakStmt(VALID_PTR(pszBackend) && *pszBackend,
5398 ("pszBackend=%#p \"%s\"\n", pszBackend, pszBackend),
5399 rc = VERR_INVALID_PARAMETER);
5400 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
5401 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
5402 rc = VERR_INVALID_PARAMETER);
5403 AssertMsgBreakStmt(cbSize,
5404 ("cbSize=%llu\n", cbSize),
5405 rc = VERR_INVALID_PARAMETER);
5406 AssertMsgBreakStmt((uImageFlags & ~VD_IMAGE_FLAGS_MASK) == 0,
5407 ("uImageFlags=%#x\n", uImageFlags),
5408 rc = VERR_INVALID_PARAMETER);
5409 /* The UUID may be NULL. */
5410 AssertMsgBreakStmt(pUuid == NULL || VALID_PTR(pUuid),
5411 ("pUuid=%#p UUID=%RTuuid\n", pUuid, pUuid),
5412 rc = VERR_INVALID_PARAMETER);
5413 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
5414 ("uOpenFlags=%#x\n", uOpenFlags),
5415 rc = VERR_INVALID_PARAMETER);
5416
5417 /* Check state. Needs a temporary read lock. Holding the write lock
5418 * all the time would be blocking other activities for too long. */
5419 rc2 = vdThreadStartRead(pDisk);
5420 AssertRC(rc2);
5421 fLockRead = true;
5422 AssertMsgBreakStmt(!pDisk->pCache,
5423 ("Create cache image cannot be done with a cache already attached\n"),
5424 rc = VERR_VD_CACHE_ALREADY_EXISTS);
5425 rc2 = vdThreadFinishRead(pDisk);
5426 AssertRC(rc2);
5427 fLockRead = false;
5428
5429 /* Set up image descriptor. */
5430 pCache = (PVDCACHE)RTMemAllocZ(sizeof(VDCACHE));
5431 if (!pCache)
5432 {
5433 rc = VERR_NO_MEMORY;
5434 break;
5435 }
5436 pCache->pszFilename = RTStrDup(pszFilename);
5437 if (!pCache->pszFilename)
5438 {
5439 rc = VERR_NO_MEMORY;
5440 break;
5441 }
5442
5443 rc = vdFindCacheBackend(pszBackend, &pCache->Backend);
5444 if (RT_FAILURE(rc))
5445 break;
5446 if (!pCache->Backend)
5447 {
5448 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
5449 N_("VD: unknown backend name '%s'"), pszBackend);
5450 break;
5451 }
5452
5453 pCache->VDIo.pDisk = pDisk;
5454 pCache->pVDIfsCache = pVDIfsCache;
5455
5456 /* Set up the I/O interface. */
5457 pCache->VDIo.pInterfaceIO = VDInterfaceGet(pVDIfsCache, VDINTERFACETYPE_IO);
5458 if (pCache->VDIo.pInterfaceIO)
5459 pCache->VDIo.pInterfaceIOCallbacks = VDGetInterfaceIO(pCache->VDIo.pInterfaceIO);
5460 else
5461 {
5462 rc = VDInterfaceAdd(&pCache->VDIo.VDIIO, "VD_IO", VDINTERFACETYPE_IO,
5463 &pDisk->VDIIOCallbacks, pDisk, &pVDIfsCache);
5464 pCache->VDIo.pInterfaceIO = &pCache->VDIo.VDIIO;
5465 pCache->VDIo.pInterfaceIOCallbacks = &pDisk->VDIIOCallbacks;
5466 }
5467
5468 /* Set up the internal I/O interface. */
5469 AssertBreakStmt(!VDInterfaceGet(pVDIfsCache, VDINTERFACETYPE_IOINT),
5470 rc = VERR_INVALID_PARAMETER);
5471 rc = VDInterfaceAdd(&pCache->VDIo.VDIIOInt, "VD_IOINT", VDINTERFACETYPE_IOINT,
5472 &pDisk->VDIIOIntCallbacks, &pCache->VDIo, &pCache->pVDIfsCache);
5473 AssertRC(rc);
5474
5475 /* Create UUID if the caller didn't specify one. */
5476 if (!pUuid)
5477 {
5478 rc = RTUuidCreate(&uuid);
5479 if (RT_FAILURE(rc))
5480 {
5481 rc = vdError(pDisk, rc, RT_SRC_POS,
5482 N_("VD: cannot generate UUID for image '%s'"),
5483 pszFilename);
5484 break;
5485 }
5486 pUuid = &uuid;
5487 }
5488
5489 pCache->uOpenFlags = uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME;
5490 rc = pCache->Backend->pfnCreate(pCache->pszFilename, cbSize,
5491 uImageFlags,
5492 pszComment, pUuid,
5493 uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME,
5494 0, 99,
5495 pDisk->pVDIfsDisk,
5496 pCache->pVDIfsCache,
5497 pVDIfsOperation,
5498 &pCache->pBackendData);
5499
5500 if (RT_SUCCESS(rc))
5501 {
5502 /* Lock disk for writing, as we modify pDisk information below. */
5503 rc2 = vdThreadStartWrite(pDisk);
5504 AssertRC(rc2);
5505 fLockWrite = true;
5506
5507 pCache->VDIo.pBackendData = pCache->pBackendData;
5508
5509 /* Re-check state, as the lock wasn't held and another image
5510 * creation call could have been done by another thread. */
5511 AssertMsgStmt(!pDisk->pCache,
5512 ("Create cache image cannot be done with another cache open\n"),
5513 rc = VERR_VD_CACHE_ALREADY_EXISTS);
5514 }
5515
5516 if ( RT_SUCCESS(rc)
5517 && pDisk->pLast)
5518 {
5519 RTUUID UuidModification;
5520
5521 /* Set same modification Uuid as the last image. */
5522 rc = pDisk->pLast->Backend->pfnGetModificationUuid(pDisk->pLast->pBackendData,
5523 &UuidModification);
5524 if (RT_SUCCESS(rc))
5525 {
5526 rc = pCache->Backend->pfnSetModificationUuid(pCache->pBackendData,
5527 &UuidModification);
5528 }
5529
5530 if (rc == VERR_NOT_SUPPORTED)
5531 rc = VINF_SUCCESS;
5532 }
5533
5534 if (RT_SUCCESS(rc))
5535 {
5536 /* Cache successfully created. */
5537 pDisk->pCache = pCache;
5538 }
5539 else
5540 {
5541 /* Error detected, but image opened. Close and delete image. */
5542 rc2 = pCache->Backend->pfnClose(pCache->pBackendData, true);
5543 AssertRC(rc2);
5544 pCache->pBackendData = NULL;
5545 }
5546 } while (0);
5547
5548 if (RT_UNLIKELY(fLockWrite))
5549 {
5550 rc2 = vdThreadFinishWrite(pDisk);
5551 AssertRC(rc2);
5552 }
5553 else if (RT_UNLIKELY(fLockRead))
5554 {
5555 rc2 = vdThreadFinishRead(pDisk);
5556 AssertRC(rc2);
5557 }
5558
5559 if (RT_FAILURE(rc))
5560 {
5561 if (pCache)
5562 {
5563 if (pCache->pszFilename)
5564 RTStrFree(pCache->pszFilename);
5565 RTMemFree(pCache);
5566 }
5567 }
5568
5569 if (RT_SUCCESS(rc) && pCbProgress && pCbProgress->pfnProgress)
5570 pCbProgress->pfnProgress(pIfProgress->pvUser, 100);
5571
5572 LogFlowFunc(("returns %Rrc\n", rc));
5573 return rc;
5574}
5575
5576/**
5577 * Merges two images (not necessarily with direct parent/child relationship).
5578 * As a side effect the source image and potentially the other images which
5579 * are also merged to the destination are deleted from both the disk and the
5580 * images in the HDD container.
5581 *
5582 * @returns VBox status code.
5583 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
5584 * @param pDisk Pointer to HDD container.
5585 * @param nImageFrom Name of the image file to merge from.
5586 * @param nImageTo Name of the image file to merge to.
5587 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
5588 */
5589VBOXDDU_DECL(int) VDMerge(PVBOXHDD pDisk, unsigned nImageFrom,
5590 unsigned nImageTo, PVDINTERFACE pVDIfsOperation)
5591{
5592 int rc = VINF_SUCCESS;
5593 int rc2;
5594 bool fLockWrite = false, fLockRead = false;
5595 void *pvBuf = NULL;
5596
5597 LogFlowFunc(("pDisk=%#p nImageFrom=%u nImageTo=%u pVDIfsOperation=%#p\n",
5598 pDisk, nImageFrom, nImageTo, pVDIfsOperation));
5599
5600 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
5601 VDINTERFACETYPE_PROGRESS);
5602 PVDINTERFACEPROGRESS pCbProgress = NULL;
5603 if (pIfProgress)
5604 pCbProgress = VDGetInterfaceProgress(pIfProgress);
5605
5606 do
5607 {
5608 /* sanity check */
5609 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
5610 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
5611
5612 /* For simplicity reasons lock for writing as the image reopen below
5613 * might need it. After all the reopen is usually needed. */
5614 rc2 = vdThreadStartWrite(pDisk);
5615 AssertRC(rc2);
5616 fLockWrite = true;
5617 PVDIMAGE pImageFrom = vdGetImageByNumber(pDisk, nImageFrom);
5618 PVDIMAGE pImageTo = vdGetImageByNumber(pDisk, nImageTo);
5619 if (!pImageFrom || !pImageTo)
5620 {
5621 rc = VERR_VD_IMAGE_NOT_FOUND;
5622 break;
5623 }
5624 AssertBreakStmt(pImageFrom != pImageTo, rc = VERR_INVALID_PARAMETER);
5625
5626 /* Make sure destination image is writable. */
5627 unsigned uOpenFlags = pImageTo->Backend->pfnGetOpenFlags(pImageTo->pBackendData);
5628 if (uOpenFlags & VD_OPEN_FLAGS_READONLY)
5629 {
5630 uOpenFlags &= ~VD_OPEN_FLAGS_READONLY;
5631 rc = pImageTo->Backend->pfnSetOpenFlags(pImageTo->pBackendData,
5632 uOpenFlags);
5633 if (RT_FAILURE(rc))
5634 break;
5635 }
5636
5637 /* Get size of destination image. */
5638 uint64_t cbSize = pImageTo->Backend->pfnGetSize(pImageTo->pBackendData);
5639 rc2 = vdThreadFinishWrite(pDisk);
5640 AssertRC(rc2);
5641 fLockWrite = false;
5642
5643 /* Allocate tmp buffer. */
5644 pvBuf = RTMemTmpAlloc(VD_MERGE_BUFFER_SIZE);
5645 if (!pvBuf)
5646 {
5647 rc = VERR_NO_MEMORY;
5648 break;
5649 }
5650
5651 /* Merging is done directly on the images itself. This potentially
5652 * causes trouble if the disk is full in the middle of operation. */
5653 if (nImageFrom < nImageTo)
5654 {
5655 /* Merge parent state into child. This means writing all not
5656 * allocated blocks in the destination image which are allocated in
5657 * the images to be merged. */
5658 uint64_t uOffset = 0;
5659 uint64_t cbRemaining = cbSize;
5660 do
5661 {
5662 size_t cbThisRead = RT_MIN(VD_MERGE_BUFFER_SIZE, cbRemaining);
5663
5664 /* Need to hold the write lock during a read-write operation. */
5665 rc2 = vdThreadStartWrite(pDisk);
5666 AssertRC(rc2);
5667 fLockWrite = true;
5668
5669 rc = pImageTo->Backend->pfnRead(pImageTo->pBackendData,
5670 uOffset, pvBuf, cbThisRead,
5671 &cbThisRead);
5672 if (rc == VERR_VD_BLOCK_FREE)
5673 {
5674 /* Search for image with allocated block. Do not attempt to
5675 * read more than the previous reads marked as valid.
5676 * Otherwise this would return stale data when different
5677 * block sizes are used for the images. */
5678 for (PVDIMAGE pCurrImage = pImageTo->pPrev;
5679 pCurrImage != NULL && pCurrImage != pImageFrom->pPrev && rc == VERR_VD_BLOCK_FREE;
5680 pCurrImage = pCurrImage->pPrev)
5681 {
5682 rc = pCurrImage->Backend->pfnRead(pCurrImage->pBackendData,
5683 uOffset, pvBuf,
5684 cbThisRead,
5685 &cbThisRead);
5686 }
5687
5688 if (rc != VERR_VD_BLOCK_FREE)
5689 {
5690 if (RT_FAILURE(rc))
5691 break;
5692 /* Updating the cache is required because this might be a live merge. */
5693 rc = vdWriteHelperEx(pDisk, pImageTo, pImageFrom->pPrev,
5694 uOffset, pvBuf, cbThisRead,
5695 true /* fUpdateCache */, 0);
5696 if (RT_FAILURE(rc))
5697 break;
5698 }
5699 else
5700 rc = VINF_SUCCESS;
5701 }
5702 else if (RT_FAILURE(rc))
5703 break;
5704
5705 rc2 = vdThreadFinishWrite(pDisk);
5706 AssertRC(rc2);
5707 fLockWrite = false;
5708
5709 uOffset += cbThisRead;
5710 cbRemaining -= cbThisRead;
5711
5712 if (pCbProgress && pCbProgress->pfnProgress)
5713 {
5714 /** @todo r=klaus: this can update the progress to the same
5715 * percentage over and over again if the image format makes
5716 * relatively small increments. */
5717 rc = pCbProgress->pfnProgress(pIfProgress->pvUser,
5718 uOffset * 99 / cbSize);
5719 if (RT_FAILURE(rc))
5720 break;
5721 }
5722 } while (uOffset < cbSize);
5723 }
5724 else
5725 {
5726 /*
5727 * We may need to update the parent uuid of the child coming after
5728 * the last image to be merged. We have to reopen it read/write.
5729 *
5730 * This is done before we do the actual merge to prevent an
5731 * inconsistent chain if the mode change fails for some reason.
5732 */
5733 if (pImageFrom->pNext)
5734 {
5735 PVDIMAGE pImageChild = pImageFrom->pNext;
5736
5737 /* Take the write lock. */
5738 rc2 = vdThreadStartWrite(pDisk);
5739 AssertRC(rc2);
5740 fLockWrite = true;
5741
5742 /* We need to open the image in read/write mode. */
5743 uOpenFlags = pImageChild->Backend->pfnGetOpenFlags(pImageChild->pBackendData);
5744
5745 if (uOpenFlags & VD_OPEN_FLAGS_READONLY)
5746 {
5747 uOpenFlags &= ~VD_OPEN_FLAGS_READONLY;
5748 rc = pImageChild->Backend->pfnSetOpenFlags(pImageChild->pBackendData,
5749 uOpenFlags);
5750 if (RT_FAILURE(rc))
5751 break;
5752 }
5753
5754 rc2 = vdThreadFinishWrite(pDisk);
5755 AssertRC(rc2);
5756 fLockWrite = false;
5757 }
5758
5759 /* If the merge is from the last image we have to relay all writes
5760 * to the merge destination as well, so that concurrent writes
5761 * (in case of a live merge) are handled correctly. */
5762 if (!pImageFrom->pNext)
5763 {
5764 /* Take the write lock. */
5765 rc2 = vdThreadStartWrite(pDisk);
5766 AssertRC(rc2);
5767 fLockWrite = true;
5768
5769 pDisk->pImageRelay = pImageTo;
5770
5771 rc2 = vdThreadFinishWrite(pDisk);
5772 AssertRC(rc2);
5773 fLockWrite = false;
5774 }
5775
5776 /* Merge child state into parent. This means writing all blocks
5777 * which are allocated in the image up to the source image to the
5778 * destination image. */
5779 uint64_t uOffset = 0;
5780 uint64_t cbRemaining = cbSize;
5781 do
5782 {
5783 size_t cbThisRead = RT_MIN(VD_MERGE_BUFFER_SIZE, cbRemaining);
5784 rc = VERR_VD_BLOCK_FREE;
5785
5786 /* Need to hold the write lock during a read-write operation. */
5787 rc2 = vdThreadStartWrite(pDisk);
5788 AssertRC(rc2);
5789 fLockWrite = true;
5790
5791 /* Search for image with allocated block. Do not attempt to
5792 * read more than the previous reads marked as valid. Otherwise
5793 * this would return stale data when different block sizes are
5794 * used for the images. */
5795 for (PVDIMAGE pCurrImage = pImageFrom;
5796 pCurrImage != NULL && pCurrImage != pImageTo && rc == VERR_VD_BLOCK_FREE;
5797 pCurrImage = pCurrImage->pPrev)
5798 {
5799 rc = pCurrImage->Backend->pfnRead(pCurrImage->pBackendData,
5800 uOffset, pvBuf,
5801 cbThisRead, &cbThisRead);
5802 }
5803
5804 if (rc != VERR_VD_BLOCK_FREE)
5805 {
5806 if (RT_FAILURE(rc))
5807 break;
5808 rc = vdWriteHelper(pDisk, pImageTo, uOffset, pvBuf,
5809 cbThisRead, true /* fUpdateCache */);
5810 if (RT_FAILURE(rc))
5811 break;
5812 }
5813 else
5814 rc = VINF_SUCCESS;
5815
5816 rc2 = vdThreadFinishWrite(pDisk);
5817 AssertRC(rc2);
5818 fLockWrite = false;
5819
5820 uOffset += cbThisRead;
5821 cbRemaining -= cbThisRead;
5822
5823 if (pCbProgress && pCbProgress->pfnProgress)
5824 {
5825 /** @todo r=klaus: this can update the progress to the same
5826 * percentage over and over again if the image format makes
5827 * relatively small increments. */
5828 rc = pCbProgress->pfnProgress(pIfProgress->pvUser,
5829 uOffset * 99 / cbSize);
5830 if (RT_FAILURE(rc))
5831 break;
5832 }
5833 } while (uOffset < cbSize);
5834
5835 /* In case we set up a "write proxy" image above we must clear
5836 * this again now to prevent stray writes. Failure or not. */
5837 if (!pImageFrom->pNext)
5838 {
5839 /* Take the write lock. */
5840 rc2 = vdThreadStartWrite(pDisk);
5841 AssertRC(rc2);
5842 fLockWrite = true;
5843
5844 pDisk->pImageRelay = NULL;
5845
5846 rc2 = vdThreadFinishWrite(pDisk);
5847 AssertRC(rc2);
5848 fLockWrite = false;
5849 }
5850 }
5851
5852 /*
5853 * Leave in case of an error to avoid corrupted data in the image chain
5854 * (includes cancelling the operation by the user).
5855 */
5856 if (RT_FAILURE(rc))
5857 break;
5858
5859 /* Need to hold the write lock while finishing the merge. */
5860 rc2 = vdThreadStartWrite(pDisk);
5861 AssertRC(rc2);
5862 fLockWrite = true;
5863
5864 /* Update parent UUID so that image chain is consistent. */
5865 RTUUID Uuid;
5866 PVDIMAGE pImageChild = NULL;
5867 if (nImageFrom < nImageTo)
5868 {
5869 if (pImageFrom->pPrev)
5870 {
5871 rc = pImageFrom->pPrev->Backend->pfnGetUuid(pImageFrom->pPrev->pBackendData,
5872 &Uuid);
5873 AssertRC(rc);
5874 }
5875 else
5876 RTUuidClear(&Uuid);
5877 rc = pImageTo->Backend->pfnSetParentUuid(pImageTo->pBackendData,
5878 &Uuid);
5879 AssertRC(rc);
5880 }
5881 else
5882 {
5883 /* Update the parent uuid of the child of the last merged image. */
5884 if (pImageFrom->pNext)
5885 {
5886 rc = pImageTo->Backend->pfnGetUuid(pImageTo->pBackendData,
5887 &Uuid);
5888 AssertRC(rc);
5889
5890 rc = pImageFrom->Backend->pfnSetParentUuid(pImageFrom->pNext->pBackendData,
5891 &Uuid);
5892 AssertRC(rc);
5893
5894 pImageChild = pImageFrom->pNext;
5895 }
5896 }
5897
5898 /* Delete the no longer needed images. */
5899 PVDIMAGE pImg = pImageFrom, pTmp;
5900 while (pImg != pImageTo)
5901 {
5902 if (nImageFrom < nImageTo)
5903 pTmp = pImg->pNext;
5904 else
5905 pTmp = pImg->pPrev;
5906 vdRemoveImageFromList(pDisk, pImg);
5907 pImg->Backend->pfnClose(pImg->pBackendData, true);
5908 RTMemFree(pImg->pszFilename);
5909 RTMemFree(pImg);
5910 pImg = pTmp;
5911 }
5912
5913 /* Make sure destination image is back to read only if necessary. */
5914 if (pImageTo != pDisk->pLast)
5915 {
5916 uOpenFlags = pImageTo->Backend->pfnGetOpenFlags(pImageTo->pBackendData);
5917 uOpenFlags |= VD_OPEN_FLAGS_READONLY;
5918 rc = pImageTo->Backend->pfnSetOpenFlags(pImageTo->pBackendData,
5919 uOpenFlags);
5920 if (RT_FAILURE(rc))
5921 break;
5922 }
5923
5924 /*
5925 * Make sure the child is readonly
5926 * for the child -> parent merge direction
5927 * if necessary.
5928 */
5929 if ( nImageFrom > nImageTo
5930 && pImageChild
5931 && pImageChild != pDisk->pLast)
5932 {
5933 uOpenFlags = pImageChild->Backend->pfnGetOpenFlags(pImageChild->pBackendData);
5934 uOpenFlags |= VD_OPEN_FLAGS_READONLY;
5935 rc = pImageChild->Backend->pfnSetOpenFlags(pImageChild->pBackendData,
5936 uOpenFlags);
5937 if (RT_FAILURE(rc))
5938 break;
5939 }
5940 } while (0);
5941
5942 if (RT_UNLIKELY(fLockWrite))
5943 {
5944 rc2 = vdThreadFinishWrite(pDisk);
5945 AssertRC(rc2);
5946 }
5947 else if (RT_UNLIKELY(fLockRead))
5948 {
5949 rc2 = vdThreadFinishRead(pDisk);
5950 AssertRC(rc2);
5951 }
5952
5953 if (pvBuf)
5954 RTMemTmpFree(pvBuf);
5955
5956 if (RT_SUCCESS(rc) && pCbProgress && pCbProgress->pfnProgress)
5957 pCbProgress->pfnProgress(pIfProgress->pvUser, 100);
5958
5959 LogFlowFunc(("returns %Rrc\n", rc));
5960 return rc;
5961}
5962
5963/**
5964 * Copies an image from one HDD container to another - extended version.
5965 * The copy is opened in the target HDD container.
5966 * It is possible to convert between different image formats, because the
5967 * backend for the destination may be different from the source.
5968 * If both the source and destination reference the same HDD container,
5969 * then the image is moved (by copying/deleting or renaming) to the new location.
5970 * The source container is unchanged if the move operation fails, otherwise
5971 * the image at the new location is opened in the same way as the old one was.
5972 *
5973 * @note The read/write accesses across disks are not synchronized, just the
5974 * accesses to each disk. Once there is a use case which requires a defined
5975 * read/write behavior in this situation this needs to be extended.
5976 *
5977 * @return VBox status code.
5978 * @return VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
5979 * @param pDiskFrom Pointer to source HDD container.
5980 * @param nImage Image number, counts from 0. 0 is always base image of container.
5981 * @param pDiskTo Pointer to destination HDD container.
5982 * @param pszBackend Name of the image file backend to use (may be NULL to use the same as the source, case insensitive).
5983 * @param pszFilename New name of the image (may be NULL to specify that the
5984 * copy destination is the destination container, or
5985 * if pDiskFrom == pDiskTo, i.e. when moving).
5986 * @param fMoveByRename If true, attempt to perform a move by renaming (if successful the new size is ignored).
5987 * @param cbSize New image size (0 means leave unchanged).
5988 * @param nImageSameFrom todo
5989 * @param nImageSameTo todo
5990 * @param uImageFlags Flags specifying special destination image features.
5991 * @param pDstUuid New UUID of the destination image. If NULL, a new UUID is created.
5992 * This parameter is used if and only if a true copy is created.
5993 * In all rename/move cases or copy to existing image cases the modification UUIDs are copied over.
5994 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
5995 * Only used if the destination image is created.
5996 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
5997 * @param pDstVDIfsImage Pointer to the per-image VD interface list, for the
5998 * destination image.
5999 * @param pDstVDIfsOperation Pointer to the per-operation VD interface list,
6000 * for the destination operation.
6001 */
6002VBOXDDU_DECL(int) VDCopyEx(PVBOXHDD pDiskFrom, unsigned nImage, PVBOXHDD pDiskTo,
6003 const char *pszBackend, const char *pszFilename,
6004 bool fMoveByRename, uint64_t cbSize,
6005 unsigned nImageFromSame, unsigned nImageToSame,
6006 unsigned uImageFlags, PCRTUUID pDstUuid,
6007 unsigned uOpenFlags, PVDINTERFACE pVDIfsOperation,
6008 PVDINTERFACE pDstVDIfsImage,
6009 PVDINTERFACE pDstVDIfsOperation)
6010{
6011 int rc = VINF_SUCCESS;
6012 int rc2;
6013 bool fLockReadFrom = false, fLockWriteFrom = false, fLockWriteTo = false;
6014 PVDIMAGE pImageTo = NULL;
6015
6016 LogFlowFunc(("pDiskFrom=%#p nImage=%u pDiskTo=%#p pszBackend=\"%s\" pszFilename=\"%s\" fMoveByRename=%d cbSize=%llu nImageFromSame=%u nImageToSame=%u uImageFlags=%#x pDstUuid=%#p uOpenFlags=%#x pVDIfsOperation=%#p pDstVDIfsImage=%#p pDstVDIfsOperation=%#p\n",
6017 pDiskFrom, nImage, pDiskTo, pszBackend, pszFilename, fMoveByRename, cbSize, nImageFromSame, nImageToSame, uImageFlags, pDstUuid, uOpenFlags, pVDIfsOperation, pDstVDIfsImage, pDstVDIfsOperation));
6018
6019 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
6020 VDINTERFACETYPE_PROGRESS);
6021 PVDINTERFACEPROGRESS pCbProgress = NULL;
6022 if (pIfProgress)
6023 pCbProgress = VDGetInterfaceProgress(pIfProgress);
6024
6025 PVDINTERFACE pDstIfProgress = VDInterfaceGet(pDstVDIfsOperation,
6026 VDINTERFACETYPE_PROGRESS);
6027 PVDINTERFACEPROGRESS pDstCbProgress = NULL;
6028 if (pDstIfProgress)
6029 pDstCbProgress = VDGetInterfaceProgress(pDstIfProgress);
6030
6031 do {
6032 /* Check arguments. */
6033 AssertMsgBreakStmt(VALID_PTR(pDiskFrom), ("pDiskFrom=%#p\n", pDiskFrom),
6034 rc = VERR_INVALID_PARAMETER);
6035 AssertMsg(pDiskFrom->u32Signature == VBOXHDDDISK_SIGNATURE,
6036 ("u32Signature=%08x\n", pDiskFrom->u32Signature));
6037
6038 rc2 = vdThreadStartRead(pDiskFrom);
6039 AssertRC(rc2);
6040 fLockReadFrom = true;
6041 PVDIMAGE pImageFrom = vdGetImageByNumber(pDiskFrom, nImage);
6042 AssertPtrBreakStmt(pImageFrom, rc = VERR_VD_IMAGE_NOT_FOUND);
6043 AssertMsgBreakStmt(VALID_PTR(pDiskTo), ("pDiskTo=%#p\n", pDiskTo),
6044 rc = VERR_INVALID_PARAMETER);
6045 AssertMsg(pDiskTo->u32Signature == VBOXHDDDISK_SIGNATURE,
6046 ("u32Signature=%08x\n", pDiskTo->u32Signature));
6047 AssertMsgBreakStmt( (nImageFromSame < nImage || nImageFromSame == VD_IMAGE_CONTENT_UNKNOWN)
6048 && (nImageToSame < pDiskTo->cImages || nImageToSame == VD_IMAGE_CONTENT_UNKNOWN)
6049 && ( (nImageFromSame == VD_IMAGE_CONTENT_UNKNOWN && nImageToSame == VD_IMAGE_CONTENT_UNKNOWN)
6050 || (nImageFromSame != VD_IMAGE_CONTENT_UNKNOWN && nImageToSame != VD_IMAGE_CONTENT_UNKNOWN)),
6051 ("nImageFromSame=%u nImageToSame=%u\n", nImageFromSame, nImageToSame),
6052 rc = VERR_INVALID_PARAMETER);
6053
6054 /* Move the image. */
6055 if (pDiskFrom == pDiskTo)
6056 {
6057 /* Rename only works when backends are the same, are file based
6058 * and the rename method is implemented. */
6059 if ( fMoveByRename
6060 && !RTStrICmp(pszBackend, pImageFrom->Backend->pszBackendName)
6061 && pImageFrom->Backend->uBackendCaps & VD_CAP_FILE
6062 && pImageFrom->Backend->pfnRename)
6063 {
6064 rc2 = vdThreadFinishRead(pDiskFrom);
6065 AssertRC(rc2);
6066 fLockReadFrom = false;
6067
6068 rc2 = vdThreadStartWrite(pDiskFrom);
6069 AssertRC(rc2);
6070 fLockWriteFrom = true;
6071 rc = pImageFrom->Backend->pfnRename(pImageFrom->pBackendData, pszFilename ? pszFilename : pImageFrom->pszFilename);
6072 break;
6073 }
6074
6075 /** @todo Moving (including shrinking/growing) of the image is
6076 * requested, but the rename attempt failed or it wasn't possible.
6077 * Must now copy image to temp location. */
6078 AssertReleaseMsgFailed(("VDCopy: moving by copy/delete not implemented\n"));
6079 }
6080
6081 /* pszFilename is allowed to be NULL, as this indicates copy to the existing image. */
6082 AssertMsgBreakStmt(pszFilename == NULL || (VALID_PTR(pszFilename) && *pszFilename),
6083 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
6084 rc = VERR_INVALID_PARAMETER);
6085
6086 uint64_t cbSizeFrom;
6087 cbSizeFrom = pImageFrom->Backend->pfnGetSize(pImageFrom->pBackendData);
6088 if (cbSizeFrom == 0)
6089 {
6090 rc = VERR_VD_VALUE_NOT_FOUND;
6091 break;
6092 }
6093
6094 VDGEOMETRY PCHSGeometryFrom = {0, 0, 0};
6095 VDGEOMETRY LCHSGeometryFrom = {0, 0, 0};
6096 pImageFrom->Backend->pfnGetPCHSGeometry(pImageFrom->pBackendData, &PCHSGeometryFrom);
6097 pImageFrom->Backend->pfnGetLCHSGeometry(pImageFrom->pBackendData, &LCHSGeometryFrom);
6098
6099 RTUUID ImageUuid, ImageModificationUuid;
6100 if (pDiskFrom != pDiskTo)
6101 {
6102 if (pDstUuid)
6103 ImageUuid = *pDstUuid;
6104 else
6105 RTUuidCreate(&ImageUuid);
6106 }
6107 else
6108 {
6109 rc = pImageFrom->Backend->pfnGetUuid(pImageFrom->pBackendData, &ImageUuid);
6110 if (RT_FAILURE(rc))
6111 RTUuidCreate(&ImageUuid);
6112 }
6113 rc = pImageFrom->Backend->pfnGetModificationUuid(pImageFrom->pBackendData, &ImageModificationUuid);
6114 if (RT_FAILURE(rc))
6115 RTUuidClear(&ImageModificationUuid);
6116
6117 char szComment[1024];
6118 rc = pImageFrom->Backend->pfnGetComment(pImageFrom->pBackendData, szComment, sizeof(szComment));
6119 if (RT_FAILURE(rc))
6120 szComment[0] = '\0';
6121 else
6122 szComment[sizeof(szComment) - 1] = '\0';
6123
6124 rc2 = vdThreadFinishRead(pDiskFrom);
6125 AssertRC(rc2);
6126 fLockReadFrom = false;
6127
6128 rc2 = vdThreadStartRead(pDiskTo);
6129 AssertRC(rc2);
6130 unsigned cImagesTo = pDiskTo->cImages;
6131 rc2 = vdThreadFinishRead(pDiskTo);
6132 AssertRC(rc2);
6133
6134 if (pszFilename)
6135 {
6136 if (cbSize == 0)
6137 cbSize = cbSizeFrom;
6138
6139 /* Create destination image with the properties of source image. */
6140 /** @todo replace the VDCreateDiff/VDCreateBase calls by direct
6141 * calls to the backend. Unifies the code and reduces the API
6142 * dependencies. Would also make the synchronization explicit. */
6143 if (cImagesTo > 0)
6144 {
6145 rc = VDCreateDiff(pDiskTo, pszBackend, pszFilename,
6146 uImageFlags, szComment, &ImageUuid,
6147 NULL /* pParentUuid */,
6148 uOpenFlags & ~VD_OPEN_FLAGS_READONLY,
6149 pDstVDIfsImage, NULL);
6150
6151 rc2 = vdThreadStartWrite(pDiskTo);
6152 AssertRC(rc2);
6153 fLockWriteTo = true;
6154 } else {
6155 /** @todo hack to force creation of a fixed image for
6156 * the RAW backend, which can't handle anything else. */
6157 if (!RTStrICmp(pszBackend, "RAW"))
6158 uImageFlags |= VD_IMAGE_FLAGS_FIXED;
6159
6160 vdFixupPCHSGeometry(&PCHSGeometryFrom, cbSize);
6161 vdFixupLCHSGeometry(&LCHSGeometryFrom, cbSize);
6162
6163 rc = VDCreateBase(pDiskTo, pszBackend, pszFilename, cbSize,
6164 uImageFlags, szComment,
6165 &PCHSGeometryFrom, &LCHSGeometryFrom,
6166 NULL, uOpenFlags & ~VD_OPEN_FLAGS_READONLY,
6167 pDstVDIfsImage, NULL);
6168
6169 rc2 = vdThreadStartWrite(pDiskTo);
6170 AssertRC(rc2);
6171 fLockWriteTo = true;
6172
6173 if (RT_SUCCESS(rc) && !RTUuidIsNull(&ImageUuid))
6174 pDiskTo->pLast->Backend->pfnSetUuid(pDiskTo->pLast->pBackendData, &ImageUuid);
6175 }
6176 if (RT_FAILURE(rc))
6177 break;
6178
6179 pImageTo = pDiskTo->pLast;
6180 AssertPtrBreakStmt(pImageTo, rc = VERR_VD_IMAGE_NOT_FOUND);
6181
6182 cbSize = RT_MIN(cbSize, cbSizeFrom);
6183 }
6184 else
6185 {
6186 pImageTo = pDiskTo->pLast;
6187 AssertPtrBreakStmt(pImageTo, rc = VERR_VD_IMAGE_NOT_FOUND);
6188
6189 uint64_t cbSizeTo;
6190 cbSizeTo = pImageTo->Backend->pfnGetSize(pImageTo->pBackendData);
6191 if (cbSizeTo == 0)
6192 {
6193 rc = VERR_VD_VALUE_NOT_FOUND;
6194 break;
6195 }
6196
6197 if (cbSize == 0)
6198 cbSize = RT_MIN(cbSizeFrom, cbSizeTo);
6199
6200 vdFixupPCHSGeometry(&PCHSGeometryFrom, cbSize);
6201 vdFixupLCHSGeometry(&LCHSGeometryFrom, cbSize);
6202
6203 /* Update the geometry in the destination image. */
6204 pImageTo->Backend->pfnSetPCHSGeometry(pImageTo->pBackendData, &PCHSGeometryFrom);
6205 pImageTo->Backend->pfnSetLCHSGeometry(pImageTo->pBackendData, &LCHSGeometryFrom);
6206 }
6207
6208 rc2 = vdThreadFinishWrite(pDiskTo);
6209 AssertRC(rc2);
6210 fLockWriteTo = false;
6211
6212 /* Whether we can take the optimized copy path (false) or not.
6213 * Don't optimize if the image existed or if it is a child image. */
6214 bool fSuppressRedundantIo = ( !(pszFilename == NULL || cImagesTo > 0)
6215 || (nImageToSame != VD_IMAGE_CONTENT_UNKNOWN));
6216 unsigned cImagesFromReadBack, cImagesToReadBack;
6217
6218 if (nImageFromSame == VD_IMAGE_CONTENT_UNKNOWN)
6219 cImagesFromReadBack = 0;
6220 else
6221 {
6222 if (nImage == VD_LAST_IMAGE)
6223 cImagesFromReadBack = pDiskFrom->cImages - nImageFromSame - 1;
6224 else
6225 cImagesFromReadBack = nImage - nImageFromSame;
6226 }
6227
6228 if (nImageToSame == VD_IMAGE_CONTENT_UNKNOWN)
6229 cImagesToReadBack = 0;
6230 else
6231 cImagesToReadBack = pDiskTo->cImages - nImageToSame - 1;
6232
6233 /* Copy the data. */
6234 rc = vdCopyHelper(pDiskFrom, pImageFrom, pDiskTo, cbSize,
6235 cImagesFromReadBack, cImagesToReadBack,
6236 fSuppressRedundantIo, pIfProgress, pCbProgress,
6237 pDstIfProgress, pDstCbProgress);
6238
6239 if (RT_SUCCESS(rc))
6240 {
6241 rc2 = vdThreadStartWrite(pDiskTo);
6242 AssertRC(rc2);
6243 fLockWriteTo = true;
6244
6245 /* Only set modification UUID if it is non-null, since the source
6246 * backend might not provide a valid modification UUID. */
6247 if (!RTUuidIsNull(&ImageModificationUuid))
6248 pImageTo->Backend->pfnSetModificationUuid(pImageTo->pBackendData, &ImageModificationUuid);
6249
6250 /* Set the requested open flags if they differ from the value
6251 * required for creating the image and copying the contents. */
6252 if ( pImageTo && pszFilename
6253 && uOpenFlags != (uOpenFlags & ~VD_OPEN_FLAGS_READONLY))
6254 rc = pImageTo->Backend->pfnSetOpenFlags(pImageTo->pBackendData,
6255 uOpenFlags);
6256 }
6257 } while (0);
6258
6259 if (RT_FAILURE(rc) && pImageTo && pszFilename)
6260 {
6261 /* Take the write lock only if it is not taken. Not worth making the
6262 * above code even more complicated. */
6263 if (RT_UNLIKELY(!fLockWriteTo))
6264 {
6265 rc2 = vdThreadStartWrite(pDiskTo);
6266 AssertRC(rc2);
6267 fLockWriteTo = true;
6268 }
6269 /* Error detected, but new image created. Remove image from list. */
6270 vdRemoveImageFromList(pDiskTo, pImageTo);
6271
6272 /* Close and delete image. */
6273 rc2 = pImageTo->Backend->pfnClose(pImageTo->pBackendData, true);
6274 AssertRC(rc2);
6275 pImageTo->pBackendData = NULL;
6276
6277 /* Free remaining resources. */
6278 if (pImageTo->pszFilename)
6279 RTStrFree(pImageTo->pszFilename);
6280
6281 RTMemFree(pImageTo);
6282 }
6283
6284 if (RT_UNLIKELY(fLockWriteTo))
6285 {
6286 rc2 = vdThreadFinishWrite(pDiskTo);
6287 AssertRC(rc2);
6288 }
6289 if (RT_UNLIKELY(fLockWriteFrom))
6290 {
6291 rc2 = vdThreadFinishWrite(pDiskFrom);
6292 AssertRC(rc2);
6293 }
6294 else if (RT_UNLIKELY(fLockReadFrom))
6295 {
6296 rc2 = vdThreadFinishRead(pDiskFrom);
6297 AssertRC(rc2);
6298 }
6299
6300 if (RT_SUCCESS(rc))
6301 {
6302 if (pCbProgress && pCbProgress->pfnProgress)
6303 pCbProgress->pfnProgress(pIfProgress->pvUser, 100);
6304 if (pDstCbProgress && pDstCbProgress->pfnProgress)
6305 pDstCbProgress->pfnProgress(pDstIfProgress->pvUser, 100);
6306 }
6307
6308 LogFlowFunc(("returns %Rrc\n", rc));
6309 return rc;
6310}
6311
6312/**
6313 * Copies an image from one HDD container to another.
6314 * The copy is opened in the target HDD container.
6315 * It is possible to convert between different image formats, because the
6316 * backend for the destination may be different from the source.
6317 * If both the source and destination reference the same HDD container,
6318 * then the image is moved (by copying/deleting or renaming) to the new location.
6319 * The source container is unchanged if the move operation fails, otherwise
6320 * the image at the new location is opened in the same way as the old one was.
6321 *
6322 * @returns VBox status code.
6323 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
6324 * @param pDiskFrom Pointer to source HDD container.
6325 * @param nImage Image number, counts from 0. 0 is always base image of container.
6326 * @param pDiskTo Pointer to destination HDD container.
6327 * @param pszBackend Name of the image file backend to use.
6328 * @param pszFilename New name of the image (may be NULL if pDiskFrom == pDiskTo).
6329 * @param fMoveByRename If true, attempt to perform a move by renaming (if successful the new size is ignored).
6330 * @param cbSize New image size (0 means leave unchanged).
6331 * @param uImageFlags Flags specifying special destination image features.
6332 * @param pDstUuid New UUID of the destination image. If NULL, a new UUID is created.
6333 * This parameter is used if and only if a true copy is created.
6334 * In all rename/move cases the UUIDs are copied over.
6335 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
6336 * Only used if the destination image is created.
6337 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
6338 * @param pDstVDIfsImage Pointer to the per-image VD interface list, for the
6339 * destination image.
6340 * @param pDstVDIfsOperation Pointer to the per-image VD interface list,
6341 * for the destination image.
6342 */
6343VBOXDDU_DECL(int) VDCopy(PVBOXHDD pDiskFrom, unsigned nImage, PVBOXHDD pDiskTo,
6344 const char *pszBackend, const char *pszFilename,
6345 bool fMoveByRename, uint64_t cbSize,
6346 unsigned uImageFlags, PCRTUUID pDstUuid,
6347 unsigned uOpenFlags, PVDINTERFACE pVDIfsOperation,
6348 PVDINTERFACE pDstVDIfsImage,
6349 PVDINTERFACE pDstVDIfsOperation)
6350{
6351 return VDCopyEx(pDiskFrom, nImage, pDiskTo, pszBackend, pszFilename, fMoveByRename,
6352 cbSize, VD_IMAGE_CONTENT_UNKNOWN, VD_IMAGE_CONTENT_UNKNOWN,
6353 uImageFlags, pDstUuid, uOpenFlags, pVDIfsOperation,
6354 pDstVDIfsImage, pDstVDIfsOperation);
6355}
6356
6357/**
6358 * Optimizes the storage consumption of an image. Typically the unused blocks
6359 * have to be wiped with zeroes to achieve a substantial reduced storage use.
6360 * Another optimization done is reordering the image blocks, which can provide
6361 * a significant performance boost, as reads and writes tend to use less random
6362 * file offsets.
6363 *
6364 * @return VBox status code.
6365 * @return VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
6366 * @return VERR_VD_IMAGE_READ_ONLY if image is not writable.
6367 * @return VERR_NOT_SUPPORTED if this kind of image can be compacted, but
6368 * the code for this isn't implemented yet.
6369 * @param pDisk Pointer to HDD container.
6370 * @param nImage Image number, counts from 0. 0 is always base image of container.
6371 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
6372 */
6373VBOXDDU_DECL(int) VDCompact(PVBOXHDD pDisk, unsigned nImage,
6374 PVDINTERFACE pVDIfsOperation)
6375{
6376 int rc = VINF_SUCCESS;
6377 int rc2;
6378 bool fLockRead = false, fLockWrite = false;
6379 void *pvBuf = NULL;
6380 void *pvTmp = NULL;
6381
6382 LogFlowFunc(("pDisk=%#p nImage=%u pVDIfsOperation=%#p\n",
6383 pDisk, nImage, pVDIfsOperation));
6384
6385 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
6386 VDINTERFACETYPE_PROGRESS);
6387 PVDINTERFACEPROGRESS pCbProgress = NULL;
6388 if (pIfProgress)
6389 pCbProgress = VDGetInterfaceProgress(pIfProgress);
6390
6391 do {
6392 /* Check arguments. */
6393 AssertMsgBreakStmt(VALID_PTR(pDisk), ("pDisk=%#p\n", pDisk),
6394 rc = VERR_INVALID_PARAMETER);
6395 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE,
6396 ("u32Signature=%08x\n", pDisk->u32Signature));
6397
6398 rc2 = vdThreadStartRead(pDisk);
6399 AssertRC(rc2);
6400 fLockRead = true;
6401
6402 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
6403 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
6404
6405 /* If there is no compact callback for not file based backends then
6406 * the backend doesn't need compaction. No need to make much fuss about
6407 * this. For file based ones signal this as not yet supported. */
6408 if (!pImage->Backend->pfnCompact)
6409 {
6410 if (pImage->Backend->uBackendCaps & VD_CAP_FILE)
6411 rc = VERR_NOT_SUPPORTED;
6412 else
6413 rc = VINF_SUCCESS;
6414 break;
6415 }
6416
6417 /* Insert interface for reading parent state into per-operation list,
6418 * if there is a parent image. */
6419 VDINTERFACE IfOpParent;
6420 VDINTERFACEPARENTSTATE ParentCb;
6421 VDPARENTSTATEDESC ParentUser;
6422 if (pImage->pPrev)
6423 {
6424 ParentCb.cbSize = sizeof(ParentCb);
6425 ParentCb.enmInterface = VDINTERFACETYPE_PARENTSTATE;
6426 ParentCb.pfnParentRead = vdParentRead;
6427 ParentUser.pDisk = pDisk;
6428 ParentUser.pImage = pImage->pPrev;
6429 rc = VDInterfaceAdd(&IfOpParent, "VDCompact_ParentState", VDINTERFACETYPE_PARENTSTATE,
6430 &ParentCb, &ParentUser, &pVDIfsOperation);
6431 AssertRC(rc);
6432 }
6433
6434 rc2 = vdThreadFinishRead(pDisk);
6435 AssertRC(rc2);
6436 fLockRead = false;
6437
6438 rc2 = vdThreadStartWrite(pDisk);
6439 AssertRC(rc2);
6440 fLockWrite = true;
6441
6442 rc = pImage->Backend->pfnCompact(pImage->pBackendData,
6443 0, 99,
6444 pDisk->pVDIfsDisk,
6445 pImage->pVDIfsImage,
6446 pVDIfsOperation);
6447 } while (0);
6448
6449 if (RT_UNLIKELY(fLockWrite))
6450 {
6451 rc2 = vdThreadFinishWrite(pDisk);
6452 AssertRC(rc2);
6453 }
6454 else if (RT_UNLIKELY(fLockRead))
6455 {
6456 rc2 = vdThreadFinishRead(pDisk);
6457 AssertRC(rc2);
6458 }
6459
6460 if (pvBuf)
6461 RTMemTmpFree(pvBuf);
6462 if (pvTmp)
6463 RTMemTmpFree(pvTmp);
6464
6465 if (RT_SUCCESS(rc))
6466 {
6467 if (pCbProgress && pCbProgress->pfnProgress)
6468 pCbProgress->pfnProgress(pIfProgress->pvUser, 100);
6469 }
6470
6471 LogFlowFunc(("returns %Rrc\n", rc));
6472 return rc;
6473}
6474
6475/**
6476 * Resizes the the given disk image to the given size.
6477 *
6478 * @return VBox status
6479 * @return VERR_VD_IMAGE_READ_ONLY if image is not writable.
6480 * @return VERR_NOT_SUPPORTED if this kind of image can be compacted, but
6481 *
6482 * @param pDisk Pointer to the HDD container.
6483 * @param cbSize New size of the image.
6484 * @param pPCHSGeometry Pointer to the new physical disk geometry <= (16383,16,63). Not NULL.
6485 * @param pLCHSGeometry Pointer to the new logical disk geometry <= (x,255,63). Not NULL.
6486 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
6487 */
6488VBOXDDU_DECL(int) VDResize(PVBOXHDD pDisk, uint64_t cbSize,
6489 PCVDGEOMETRY pPCHSGeometry,
6490 PCVDGEOMETRY pLCHSGeometry,
6491 PVDINTERFACE pVDIfsOperation)
6492{
6493 /** @todo r=klaus resizing was designed to be part of VDCopy, so having a separate function is not desirable. */
6494 int rc = VINF_SUCCESS;
6495 int rc2;
6496 bool fLockRead = false, fLockWrite = false;
6497
6498 LogFlowFunc(("pDisk=%#p cbSize=%llu pVDIfsOperation=%#p\n",
6499 pDisk, cbSize, pVDIfsOperation));
6500
6501 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
6502 VDINTERFACETYPE_PROGRESS);
6503 PVDINTERFACEPROGRESS pCbProgress = NULL;
6504 if (pIfProgress)
6505 pCbProgress = VDGetInterfaceProgress(pIfProgress);
6506
6507 do {
6508 /* Check arguments. */
6509 AssertMsgBreakStmt(VALID_PTR(pDisk), ("pDisk=%#p\n", pDisk),
6510 rc = VERR_INVALID_PARAMETER);
6511 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE,
6512 ("u32Signature=%08x\n", pDisk->u32Signature));
6513
6514 rc2 = vdThreadStartRead(pDisk);
6515 AssertRC(rc2);
6516 fLockRead = true;
6517
6518 /* Not supported if the disk has child images attached. */
6519 AssertMsgBreakStmt(pDisk->cImages == 1, ("cImages=%u\n", pDisk->cImages),
6520 rc = VERR_NOT_SUPPORTED);
6521
6522 PVDIMAGE pImage = pDisk->pBase;
6523
6524 /* If there is no compact callback for not file based backends then
6525 * the backend doesn't need compaction. No need to make much fuss about
6526 * this. For file based ones signal this as not yet supported. */
6527 if (!pImage->Backend->pfnResize)
6528 {
6529 if (pImage->Backend->uBackendCaps & VD_CAP_FILE)
6530 rc = VERR_NOT_SUPPORTED;
6531 else
6532 rc = VINF_SUCCESS;
6533 break;
6534 }
6535
6536 rc2 = vdThreadFinishRead(pDisk);
6537 AssertRC(rc2);
6538 fLockRead = false;
6539
6540 rc2 = vdThreadStartWrite(pDisk);
6541 AssertRC(rc2);
6542 fLockWrite = true;
6543
6544 VDGEOMETRY PCHSGeometryOld;
6545 VDGEOMETRY LCHSGeometryOld;
6546 PCVDGEOMETRY pPCHSGeometryNew;
6547 PCVDGEOMETRY pLCHSGeometryNew;
6548
6549 if (pPCHSGeometry->cCylinders == 0)
6550 {
6551 /* Auto-detect marker, calculate new value ourself. */
6552 rc = pImage->Backend->pfnGetPCHSGeometry(pImage->pBackendData, &PCHSGeometryOld);
6553 if (RT_SUCCESS(rc) && (PCHSGeometryOld.cCylinders != 0))
6554 PCHSGeometryOld.cCylinders = RT_MIN(cbSize / 512 / PCHSGeometryOld.cHeads / PCHSGeometryOld.cSectors, 16383);
6555 else if (rc == VERR_VD_GEOMETRY_NOT_SET)
6556 rc = VINF_SUCCESS;
6557
6558 pPCHSGeometryNew = &PCHSGeometryOld;
6559 }
6560 else
6561 pPCHSGeometryNew = pPCHSGeometry;
6562
6563 if (pLCHSGeometry->cCylinders == 0)
6564 {
6565 /* Auto-detect marker, calculate new value ourself. */
6566 rc = pImage->Backend->pfnGetLCHSGeometry(pImage->pBackendData, &LCHSGeometryOld);
6567 if (RT_SUCCESS(rc) && (LCHSGeometryOld.cCylinders != 0))
6568 LCHSGeometryOld.cCylinders = cbSize / 512 / LCHSGeometryOld.cHeads / LCHSGeometryOld.cSectors;
6569 else if (rc == VERR_VD_GEOMETRY_NOT_SET)
6570 rc = VINF_SUCCESS;
6571
6572 pLCHSGeometryNew = &LCHSGeometryOld;
6573 }
6574 else
6575 pLCHSGeometryNew = pLCHSGeometry;
6576
6577 if (RT_SUCCESS(rc))
6578 rc = pImage->Backend->pfnResize(pImage->pBackendData,
6579 cbSize,
6580 pPCHSGeometryNew,
6581 pLCHSGeometryNew,
6582 0, 99,
6583 pDisk->pVDIfsDisk,
6584 pImage->pVDIfsImage,
6585 pVDIfsOperation);
6586 } while (0);
6587
6588 if (RT_UNLIKELY(fLockWrite))
6589 {
6590 rc2 = vdThreadFinishWrite(pDisk);
6591 AssertRC(rc2);
6592 }
6593 else if (RT_UNLIKELY(fLockRead))
6594 {
6595 rc2 = vdThreadFinishRead(pDisk);
6596 AssertRC(rc2);
6597 }
6598
6599 if (RT_SUCCESS(rc))
6600 {
6601 if (pCbProgress && pCbProgress->pfnProgress)
6602 pCbProgress->pfnProgress(pIfProgress->pvUser, 100);
6603 }
6604
6605 LogFlowFunc(("returns %Rrc\n", rc));
6606 return rc;
6607}
6608
6609/**
6610 * Closes the last opened image file in HDD container.
6611 * If previous image file was opened in read-only mode (the normal case) and
6612 * the last opened image is in read-write mode then the previous image will be
6613 * reopened in read/write mode.
6614 *
6615 * @returns VBox status code.
6616 * @returns VERR_VD_NOT_OPENED if no image is opened in HDD container.
6617 * @param pDisk Pointer to HDD container.
6618 * @param fDelete If true, delete the image from the host disk.
6619 */
6620VBOXDDU_DECL(int) VDClose(PVBOXHDD pDisk, bool fDelete)
6621{
6622 int rc = VINF_SUCCESS;
6623 int rc2;
6624 bool fLockWrite = false;
6625
6626 LogFlowFunc(("pDisk=%#p fDelete=%d\n", pDisk, fDelete));
6627 do
6628 {
6629 /* sanity check */
6630 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
6631 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
6632
6633 /* Not worth splitting this up into a read lock phase and write
6634 * lock phase, as closing an image is a relatively fast operation
6635 * dominated by the part which needs the write lock. */
6636 rc2 = vdThreadStartWrite(pDisk);
6637 AssertRC(rc2);
6638 fLockWrite = true;
6639
6640 PVDIMAGE pImage = pDisk->pLast;
6641 if (!pImage)
6642 {
6643 rc = VERR_VD_NOT_OPENED;
6644 break;
6645 }
6646 unsigned uOpenFlags = pImage->Backend->pfnGetOpenFlags(pImage->pBackendData);
6647 /* Remove image from list of opened images. */
6648 vdRemoveImageFromList(pDisk, pImage);
6649 /* Close (and optionally delete) image. */
6650 rc = pImage->Backend->pfnClose(pImage->pBackendData, fDelete);
6651 /* Free remaining resources related to the image. */
6652 RTStrFree(pImage->pszFilename);
6653 RTMemFree(pImage);
6654
6655 pImage = pDisk->pLast;
6656 if (!pImage)
6657 break;
6658
6659 /* If disk was previously in read/write mode, make sure it will stay
6660 * like this (if possible) after closing this image. Set the open flags
6661 * accordingly. */
6662 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY))
6663 {
6664 uOpenFlags = pImage->Backend->pfnGetOpenFlags(pImage->pBackendData);
6665 uOpenFlags &= ~ VD_OPEN_FLAGS_READONLY;
6666 rc = pImage->Backend->pfnSetOpenFlags(pImage->pBackendData, uOpenFlags);
6667 }
6668
6669 /* Cache disk information. */
6670 pDisk->cbSize = pImage->Backend->pfnGetSize(pImage->pBackendData);
6671
6672 /* Cache PCHS geometry. */
6673 rc2 = pImage->Backend->pfnGetPCHSGeometry(pImage->pBackendData,
6674 &pDisk->PCHSGeometry);
6675 if (RT_FAILURE(rc2))
6676 {
6677 pDisk->PCHSGeometry.cCylinders = 0;
6678 pDisk->PCHSGeometry.cHeads = 0;
6679 pDisk->PCHSGeometry.cSectors = 0;
6680 }
6681 else
6682 {
6683 /* Make sure the PCHS geometry is properly clipped. */
6684 pDisk->PCHSGeometry.cCylinders = RT_MIN(pDisk->PCHSGeometry.cCylinders, 16383);
6685 pDisk->PCHSGeometry.cHeads = RT_MIN(pDisk->PCHSGeometry.cHeads, 16);
6686 pDisk->PCHSGeometry.cSectors = RT_MIN(pDisk->PCHSGeometry.cSectors, 63);
6687 }
6688
6689 /* Cache LCHS geometry. */
6690 rc2 = pImage->Backend->pfnGetLCHSGeometry(pImage->pBackendData,
6691 &pDisk->LCHSGeometry);
6692 if (RT_FAILURE(rc2))
6693 {
6694 pDisk->LCHSGeometry.cCylinders = 0;
6695 pDisk->LCHSGeometry.cHeads = 0;
6696 pDisk->LCHSGeometry.cSectors = 0;
6697 }
6698 else
6699 {
6700 /* Make sure the LCHS geometry is properly clipped. */
6701 pDisk->LCHSGeometry.cHeads = RT_MIN(pDisk->LCHSGeometry.cHeads, 255);
6702 pDisk->LCHSGeometry.cSectors = RT_MIN(pDisk->LCHSGeometry.cSectors, 63);
6703 }
6704 } while (0);
6705
6706 if (RT_UNLIKELY(fLockWrite))
6707 {
6708 rc2 = vdThreadFinishWrite(pDisk);
6709 AssertRC(rc2);
6710 }
6711
6712 LogFlowFunc(("returns %Rrc\n", rc));
6713 return rc;
6714}
6715
6716/**
6717 * Closes the currently opened cache image file in HDD container.
6718 *
6719 * @return VBox status code.
6720 * @return VERR_VD_NOT_OPENED if no cache is opened in HDD container.
6721 * @param pDisk Pointer to HDD container.
6722 * @param fDelete If true, delete the image from the host disk.
6723 */
6724VBOXDDU_DECL(int) VDCacheClose(PVBOXHDD pDisk, bool fDelete)
6725{
6726 int rc = VINF_SUCCESS;
6727 int rc2;
6728 bool fLockWrite = false;
6729 PVDCACHE pCache = NULL;
6730
6731 LogFlowFunc(("pDisk=%#p fDelete=%d\n", pDisk, fDelete));
6732
6733 do
6734 {
6735 /* sanity check */
6736 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
6737 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
6738
6739 rc2 = vdThreadStartWrite(pDisk);
6740 AssertRC(rc2);
6741 fLockWrite = true;
6742
6743 AssertPtrBreakStmt(pDisk->pCache, rc = VERR_VD_CACHE_NOT_FOUND);
6744
6745 pCache = pDisk->pCache;
6746 pDisk->pCache = NULL;
6747
6748 pCache->Backend->pfnClose(pCache->pBackendData, fDelete);
6749 if (pCache->pszFilename)
6750 RTStrFree(pCache->pszFilename);
6751 RTMemFree(pCache);
6752 } while (0);
6753
6754 if (RT_LIKELY(fLockWrite))
6755 {
6756 rc2 = vdThreadFinishWrite(pDisk);
6757 AssertRC(rc2);
6758 }
6759
6760 LogFlowFunc(("returns %Rrc\n", rc));
6761 return rc;
6762}
6763
6764/**
6765 * Closes all opened image files in HDD container.
6766 *
6767 * @returns VBox status code.
6768 * @param pDisk Pointer to HDD container.
6769 */
6770VBOXDDU_DECL(int) VDCloseAll(PVBOXHDD pDisk)
6771{
6772 int rc = VINF_SUCCESS;
6773 int rc2;
6774 bool fLockWrite = false;
6775
6776 LogFlowFunc(("pDisk=%#p\n", pDisk));
6777 do
6778 {
6779 /* sanity check */
6780 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
6781 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
6782
6783 /* Lock the entire operation. */
6784 rc2 = vdThreadStartWrite(pDisk);
6785 AssertRC(rc2);
6786 fLockWrite = true;
6787
6788 PVDCACHE pCache = pDisk->pCache;
6789 if (pCache)
6790 {
6791 rc2 = pCache->Backend->pfnClose(pCache->pBackendData, false);
6792 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
6793 rc = rc2;
6794
6795 if (pCache->pszFilename)
6796 RTStrFree(pCache->pszFilename);
6797 RTMemFree(pCache);
6798 }
6799
6800 PVDIMAGE pImage = pDisk->pLast;
6801 while (VALID_PTR(pImage))
6802 {
6803 PVDIMAGE pPrev = pImage->pPrev;
6804 /* Remove image from list of opened images. */
6805 vdRemoveImageFromList(pDisk, pImage);
6806 /* Close image. */
6807 rc2 = pImage->Backend->pfnClose(pImage->pBackendData, false);
6808 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
6809 rc = rc2;
6810 /* Free remaining resources related to the image. */
6811 RTStrFree(pImage->pszFilename);
6812 RTMemFree(pImage);
6813 pImage = pPrev;
6814 }
6815 Assert(!VALID_PTR(pDisk->pLast));
6816 } while (0);
6817
6818 if (RT_UNLIKELY(fLockWrite))
6819 {
6820 rc2 = vdThreadFinishWrite(pDisk);
6821 AssertRC(rc2);
6822 }
6823
6824 LogFlowFunc(("returns %Rrc\n", rc));
6825 return rc;
6826}
6827
6828/**
6829 * Read data from virtual HDD.
6830 *
6831 * @returns VBox status code.
6832 * @returns VERR_VD_NOT_OPENED if no image is opened in HDD container.
6833 * @param pDisk Pointer to HDD container.
6834 * @param uOffset Offset of first reading byte from start of disk.
6835 * @param pvBuf Pointer to buffer for reading data.
6836 * @param cbRead Number of bytes to read.
6837 */
6838VBOXDDU_DECL(int) VDRead(PVBOXHDD pDisk, uint64_t uOffset, void *pvBuf,
6839 size_t cbRead)
6840{
6841 int rc = VINF_SUCCESS;
6842 int rc2;
6843 bool fLockRead = false;
6844
6845 LogFlowFunc(("pDisk=%#p uOffset=%llu pvBuf=%p cbRead=%zu\n",
6846 pDisk, uOffset, pvBuf, cbRead));
6847 do
6848 {
6849 /* sanity check */
6850 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
6851 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
6852
6853 /* Check arguments. */
6854 AssertMsgBreakStmt(VALID_PTR(pvBuf),
6855 ("pvBuf=%#p\n", pvBuf),
6856 rc = VERR_INVALID_PARAMETER);
6857 AssertMsgBreakStmt(cbRead,
6858 ("cbRead=%zu\n", cbRead),
6859 rc = VERR_INVALID_PARAMETER);
6860
6861 rc2 = vdThreadStartRead(pDisk);
6862 AssertRC(rc2);
6863 fLockRead = true;
6864
6865 AssertMsgBreakStmt(uOffset + cbRead <= pDisk->cbSize,
6866 ("uOffset=%llu cbRead=%zu pDisk->cbSize=%llu\n",
6867 uOffset, cbRead, pDisk->cbSize),
6868 rc = VERR_INVALID_PARAMETER);
6869
6870 PVDIMAGE pImage = pDisk->pLast;
6871 AssertPtrBreakStmt(pImage, rc = VERR_VD_NOT_OPENED);
6872
6873 rc = vdReadHelper(pDisk, pImage, uOffset, pvBuf, cbRead,
6874 true /* fUpdateCache */);
6875 } while (0);
6876
6877 if (RT_UNLIKELY(fLockRead))
6878 {
6879 rc2 = vdThreadFinishRead(pDisk);
6880 AssertRC(rc2);
6881 }
6882
6883 LogFlowFunc(("returns %Rrc\n", rc));
6884 return rc;
6885}
6886
6887/**
6888 * Write data to virtual HDD.
6889 *
6890 * @returns VBox status code.
6891 * @returns VERR_VD_NOT_OPENED if no image is opened in HDD container.
6892 * @param pDisk Pointer to HDD container.
6893 * @param uOffset Offset of the first byte being
6894 * written from start of disk.
6895 * @param pvBuf Pointer to buffer for writing data.
6896 * @param cbWrite Number of bytes to write.
6897 */
6898VBOXDDU_DECL(int) VDWrite(PVBOXHDD pDisk, uint64_t uOffset, const void *pvBuf,
6899 size_t cbWrite)
6900{
6901 int rc = VINF_SUCCESS;
6902 int rc2;
6903 bool fLockWrite = false;
6904
6905 LogFlowFunc(("pDisk=%#p uOffset=%llu pvBuf=%p cbWrite=%zu\n",
6906 pDisk, uOffset, pvBuf, cbWrite));
6907 do
6908 {
6909 /* sanity check */
6910 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
6911 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
6912
6913 /* Check arguments. */
6914 AssertMsgBreakStmt(VALID_PTR(pvBuf),
6915 ("pvBuf=%#p\n", pvBuf),
6916 rc = VERR_INVALID_PARAMETER);
6917 AssertMsgBreakStmt(cbWrite,
6918 ("cbWrite=%zu\n", cbWrite),
6919 rc = VERR_INVALID_PARAMETER);
6920
6921 rc2 = vdThreadStartWrite(pDisk);
6922 AssertRC(rc2);
6923 fLockWrite = true;
6924
6925 AssertMsgBreakStmt(uOffset + cbWrite <= pDisk->cbSize,
6926 ("uOffset=%llu cbWrite=%zu pDisk->cbSize=%llu\n",
6927 uOffset, cbWrite, pDisk->cbSize),
6928 rc = VERR_INVALID_PARAMETER);
6929
6930 PVDIMAGE pImage = pDisk->pLast;
6931 AssertPtrBreakStmt(pImage, rc = VERR_VD_NOT_OPENED);
6932
6933 vdSetModifiedFlag(pDisk);
6934 rc = vdWriteHelper(pDisk, pImage, uOffset, pvBuf, cbWrite,
6935 true /* fUpdateCache */);
6936 if (RT_FAILURE(rc))
6937 break;
6938
6939 /* If there is a merge (in the direction towards a parent) running
6940 * concurrently then we have to also "relay" the write to this parent,
6941 * as the merge position might be already past the position where
6942 * this write is going. The "context" of the write can come from the
6943 * natural chain, since merging either already did or will take care
6944 * of the "other" content which is might be needed to fill the block
6945 * to a full allocation size. The cache doesn't need to be touched
6946 * as this write is covered by the previous one. */
6947 if (RT_UNLIKELY(pDisk->pImageRelay))
6948 rc = vdWriteHelper(pDisk, pDisk->pImageRelay, uOffset,
6949 pvBuf, cbWrite, false /* fUpdateCache */);
6950 } while (0);
6951
6952 if (RT_UNLIKELY(fLockWrite))
6953 {
6954 rc2 = vdThreadFinishWrite(pDisk);
6955 AssertRC(rc2);
6956 }
6957
6958 LogFlowFunc(("returns %Rrc\n", rc));
6959 return rc;
6960}
6961
6962/**
6963 * Make sure the on disk representation of a virtual HDD is up to date.
6964 *
6965 * @returns VBox status code.
6966 * @returns VERR_VD_NOT_OPENED if no image is opened in HDD container.
6967 * @param pDisk Pointer to HDD container.
6968 */
6969VBOXDDU_DECL(int) VDFlush(PVBOXHDD pDisk)
6970{
6971 int rc = VINF_SUCCESS;
6972 int rc2;
6973 bool fLockWrite = false;
6974
6975 LogFlowFunc(("pDisk=%#p\n", pDisk));
6976 do
6977 {
6978 /* sanity check */
6979 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
6980 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
6981
6982 rc2 = vdThreadStartWrite(pDisk);
6983 AssertRC(rc2);
6984 fLockWrite = true;
6985
6986 PVDIMAGE pImage = pDisk->pLast;
6987 AssertPtrBreakStmt(pImage, rc = VERR_VD_NOT_OPENED);
6988
6989 vdResetModifiedFlag(pDisk);
6990 rc = pImage->Backend->pfnFlush(pImage->pBackendData);
6991
6992 if ( RT_SUCCESS(rc)
6993 && pDisk->pCache)
6994 rc = pDisk->pCache->Backend->pfnFlush(pDisk->pCache->pBackendData);
6995 } while (0);
6996
6997 if (RT_UNLIKELY(fLockWrite))
6998 {
6999 rc2 = vdThreadFinishWrite(pDisk);
7000 AssertRC(rc2);
7001 }
7002
7003 LogFlowFunc(("returns %Rrc\n", rc));
7004 return rc;
7005}
7006
7007/**
7008 * Get number of opened images in HDD container.
7009 *
7010 * @returns Number of opened images for HDD container. 0 if no images have been opened.
7011 * @param pDisk Pointer to HDD container.
7012 */
7013VBOXDDU_DECL(unsigned) VDGetCount(PVBOXHDD pDisk)
7014{
7015 unsigned cImages;
7016 int rc2;
7017 bool fLockRead = false;
7018
7019 LogFlowFunc(("pDisk=%#p\n", pDisk));
7020 do
7021 {
7022 /* sanity check */
7023 AssertPtrBreakStmt(pDisk, cImages = 0);
7024 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7025
7026 rc2 = vdThreadStartRead(pDisk);
7027 AssertRC(rc2);
7028 fLockRead = true;
7029
7030 cImages = pDisk->cImages;
7031 } while (0);
7032
7033 if (RT_UNLIKELY(fLockRead))
7034 {
7035 rc2 = vdThreadFinishRead(pDisk);
7036 AssertRC(rc2);
7037 }
7038
7039 LogFlowFunc(("returns %u\n", cImages));
7040 return cImages;
7041}
7042
7043/**
7044 * Get read/write mode of HDD container.
7045 *
7046 * @returns Virtual disk ReadOnly status.
7047 * @returns true if no image is opened in HDD container.
7048 * @param pDisk Pointer to HDD container.
7049 */
7050VBOXDDU_DECL(bool) VDIsReadOnly(PVBOXHDD pDisk)
7051{
7052 bool fReadOnly;
7053 int rc2;
7054 bool fLockRead = false;
7055
7056 LogFlowFunc(("pDisk=%#p\n", pDisk));
7057 do
7058 {
7059 /* sanity check */
7060 AssertPtrBreakStmt(pDisk, fReadOnly = false);
7061 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7062
7063 rc2 = vdThreadStartRead(pDisk);
7064 AssertRC(rc2);
7065 fLockRead = true;
7066
7067 PVDIMAGE pImage = pDisk->pLast;
7068 AssertPtrBreakStmt(pImage, fReadOnly = true);
7069
7070 unsigned uOpenFlags;
7071 uOpenFlags = pDisk->pLast->Backend->pfnGetOpenFlags(pDisk->pLast->pBackendData);
7072 fReadOnly = !!(uOpenFlags & VD_OPEN_FLAGS_READONLY);
7073 } while (0);
7074
7075 if (RT_UNLIKELY(fLockRead))
7076 {
7077 rc2 = vdThreadFinishRead(pDisk);
7078 AssertRC(rc2);
7079 }
7080
7081 LogFlowFunc(("returns %d\n", fReadOnly));
7082 return fReadOnly;
7083}
7084
7085/**
7086 * Get total capacity of an image in HDD container.
7087 *
7088 * @returns Virtual disk size in bytes.
7089 * @returns 0 if no image with specified number was not opened.
7090 * @param pDisk Pointer to HDD container.
7091 * @param nImage Image number, counts from 0. 0 is always base image of container.
7092 */
7093VBOXDDU_DECL(uint64_t) VDGetSize(PVBOXHDD pDisk, unsigned nImage)
7094{
7095 uint64_t cbSize;
7096 int rc2;
7097 bool fLockRead = false;
7098
7099 LogFlowFunc(("pDisk=%#p nImage=%u\n", pDisk, nImage));
7100 do
7101 {
7102 /* sanity check */
7103 AssertPtrBreakStmt(pDisk, cbSize = 0);
7104 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7105
7106 rc2 = vdThreadStartRead(pDisk);
7107 AssertRC(rc2);
7108 fLockRead = true;
7109
7110 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7111 AssertPtrBreakStmt(pImage, cbSize = 0);
7112 cbSize = pImage->Backend->pfnGetSize(pImage->pBackendData);
7113 } while (0);
7114
7115 if (RT_UNLIKELY(fLockRead))
7116 {
7117 rc2 = vdThreadFinishRead(pDisk);
7118 AssertRC(rc2);
7119 }
7120
7121 LogFlowFunc(("returns %llu\n", cbSize));
7122 return cbSize;
7123}
7124
7125/**
7126 * Get total file size of an image in HDD container.
7127 *
7128 * @returns Virtual disk size in bytes.
7129 * @returns 0 if no image is opened in HDD container.
7130 * @param pDisk Pointer to HDD container.
7131 * @param nImage Image number, counts from 0. 0 is always base image of container.
7132 */
7133VBOXDDU_DECL(uint64_t) VDGetFileSize(PVBOXHDD pDisk, unsigned nImage)
7134{
7135 uint64_t cbSize;
7136 int rc2;
7137 bool fLockRead = false;
7138
7139 LogFlowFunc(("pDisk=%#p nImage=%u\n", pDisk, nImage));
7140 do
7141 {
7142 /* sanity check */
7143 AssertPtrBreakStmt(pDisk, cbSize = 0);
7144 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7145
7146 rc2 = vdThreadStartRead(pDisk);
7147 AssertRC(rc2);
7148 fLockRead = true;
7149
7150 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7151 AssertPtrBreakStmt(pImage, cbSize = 0);
7152 cbSize = pImage->Backend->pfnGetFileSize(pImage->pBackendData);
7153 } while (0);
7154
7155 if (RT_UNLIKELY(fLockRead))
7156 {
7157 rc2 = vdThreadFinishRead(pDisk);
7158 AssertRC(rc2);
7159 }
7160
7161 LogFlowFunc(("returns %llu\n", cbSize));
7162 return cbSize;
7163}
7164
7165/**
7166 * Get virtual disk PCHS geometry stored in HDD container.
7167 *
7168 * @returns VBox status code.
7169 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7170 * @returns VERR_VD_GEOMETRY_NOT_SET if no geometry present in the HDD container.
7171 * @param pDisk Pointer to HDD container.
7172 * @param nImage Image number, counts from 0. 0 is always base image of container.
7173 * @param pPCHSGeometry Where to store PCHS geometry. Not NULL.
7174 */
7175VBOXDDU_DECL(int) VDGetPCHSGeometry(PVBOXHDD pDisk, unsigned nImage,
7176 PVDGEOMETRY pPCHSGeometry)
7177{
7178 int rc = VINF_SUCCESS;
7179 int rc2;
7180 bool fLockRead = false;
7181
7182 LogFlowFunc(("pDisk=%#p nImage=%u pPCHSGeometry=%#p\n",
7183 pDisk, nImage, pPCHSGeometry));
7184 do
7185 {
7186 /* sanity check */
7187 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7188 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7189
7190 /* Check arguments. */
7191 AssertMsgBreakStmt(VALID_PTR(pPCHSGeometry),
7192 ("pPCHSGeometry=%#p\n", pPCHSGeometry),
7193 rc = VERR_INVALID_PARAMETER);
7194
7195 rc2 = vdThreadStartRead(pDisk);
7196 AssertRC(rc2);
7197 fLockRead = true;
7198
7199 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7200 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7201
7202 if (pImage == pDisk->pLast)
7203 {
7204 /* Use cached information if possible. */
7205 if (pDisk->PCHSGeometry.cCylinders != 0)
7206 *pPCHSGeometry = pDisk->PCHSGeometry;
7207 else
7208 rc = VERR_VD_GEOMETRY_NOT_SET;
7209 }
7210 else
7211 rc = pImage->Backend->pfnGetPCHSGeometry(pImage->pBackendData,
7212 pPCHSGeometry);
7213 } while (0);
7214
7215 if (RT_UNLIKELY(fLockRead))
7216 {
7217 rc2 = vdThreadFinishRead(pDisk);
7218 AssertRC(rc2);
7219 }
7220
7221 LogFlowFunc(("%Rrc (PCHS=%u/%u/%u)\n", rc,
7222 pDisk->PCHSGeometry.cCylinders, pDisk->PCHSGeometry.cHeads,
7223 pDisk->PCHSGeometry.cSectors));
7224 return rc;
7225}
7226
7227/**
7228 * Store virtual disk PCHS geometry in HDD container.
7229 *
7230 * Note that in case of unrecoverable error all images in HDD container will be closed.
7231 *
7232 * @returns VBox status code.
7233 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7234 * @returns VERR_VD_GEOMETRY_NOT_SET if no geometry present in the HDD container.
7235 * @param pDisk Pointer to HDD container.
7236 * @param nImage Image number, counts from 0. 0 is always base image of container.
7237 * @param pPCHSGeometry Where to load PCHS geometry from. Not NULL.
7238 */
7239VBOXDDU_DECL(int) VDSetPCHSGeometry(PVBOXHDD pDisk, unsigned nImage,
7240 PCVDGEOMETRY pPCHSGeometry)
7241{
7242 int rc = VINF_SUCCESS;
7243 int rc2;
7244 bool fLockWrite = false;
7245
7246 LogFlowFunc(("pDisk=%#p nImage=%u pPCHSGeometry=%#p PCHS=%u/%u/%u\n",
7247 pDisk, nImage, pPCHSGeometry, pPCHSGeometry->cCylinders,
7248 pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
7249 do
7250 {
7251 /* sanity check */
7252 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7253 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7254
7255 /* Check arguments. */
7256 AssertMsgBreakStmt( VALID_PTR(pPCHSGeometry)
7257 && pPCHSGeometry->cHeads <= 16
7258 && pPCHSGeometry->cSectors <= 63,
7259 ("pPCHSGeometry=%#p PCHS=%u/%u/%u\n", pPCHSGeometry,
7260 pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads,
7261 pPCHSGeometry->cSectors),
7262 rc = VERR_INVALID_PARAMETER);
7263
7264 rc2 = vdThreadStartWrite(pDisk);
7265 AssertRC(rc2);
7266 fLockWrite = true;
7267
7268 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7269 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7270
7271 if (pImage == pDisk->pLast)
7272 {
7273 if ( pPCHSGeometry->cCylinders != pDisk->PCHSGeometry.cCylinders
7274 || pPCHSGeometry->cHeads != pDisk->PCHSGeometry.cHeads
7275 || pPCHSGeometry->cSectors != pDisk->PCHSGeometry.cSectors)
7276 {
7277 /* Only update geometry if it is changed. Avoids similar checks
7278 * in every backend. Most of the time the new geometry is set
7279 * to the previous values, so no need to go through the hassle
7280 * of updating an image which could be opened in read-only mode
7281 * right now. */
7282 rc = pImage->Backend->pfnSetPCHSGeometry(pImage->pBackendData,
7283 pPCHSGeometry);
7284
7285 /* Cache new geometry values in any case. */
7286 rc2 = pImage->Backend->pfnGetPCHSGeometry(pImage->pBackendData,
7287 &pDisk->PCHSGeometry);
7288 if (RT_FAILURE(rc2))
7289 {
7290 pDisk->PCHSGeometry.cCylinders = 0;
7291 pDisk->PCHSGeometry.cHeads = 0;
7292 pDisk->PCHSGeometry.cSectors = 0;
7293 }
7294 else
7295 {
7296 /* Make sure the CHS geometry is properly clipped. */
7297 pDisk->PCHSGeometry.cHeads = RT_MIN(pDisk->PCHSGeometry.cHeads, 255);
7298 pDisk->PCHSGeometry.cSectors = RT_MIN(pDisk->PCHSGeometry.cSectors, 63);
7299 }
7300 }
7301 }
7302 else
7303 {
7304 VDGEOMETRY PCHS;
7305 rc = pImage->Backend->pfnGetPCHSGeometry(pImage->pBackendData,
7306 &PCHS);
7307 if ( RT_FAILURE(rc)
7308 || pPCHSGeometry->cCylinders != PCHS.cCylinders
7309 || pPCHSGeometry->cHeads != PCHS.cHeads
7310 || pPCHSGeometry->cSectors != PCHS.cSectors)
7311 {
7312 /* Only update geometry if it is changed. Avoids similar checks
7313 * in every backend. Most of the time the new geometry is set
7314 * to the previous values, so no need to go through the hassle
7315 * of updating an image which could be opened in read-only mode
7316 * right now. */
7317 rc = pImage->Backend->pfnSetPCHSGeometry(pImage->pBackendData,
7318 pPCHSGeometry);
7319 }
7320 }
7321 } while (0);
7322
7323 if (RT_UNLIKELY(fLockWrite))
7324 {
7325 rc2 = vdThreadFinishWrite(pDisk);
7326 AssertRC(rc2);
7327 }
7328
7329 LogFlowFunc(("returns %Rrc\n", rc));
7330 return rc;
7331}
7332
7333/**
7334 * Get virtual disk LCHS geometry stored in HDD container.
7335 *
7336 * @returns VBox status code.
7337 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7338 * @returns VERR_VD_GEOMETRY_NOT_SET if no geometry present in the HDD container.
7339 * @param pDisk Pointer to HDD container.
7340 * @param nImage Image number, counts from 0. 0 is always base image of container.
7341 * @param pLCHSGeometry Where to store LCHS geometry. Not NULL.
7342 */
7343VBOXDDU_DECL(int) VDGetLCHSGeometry(PVBOXHDD pDisk, unsigned nImage,
7344 PVDGEOMETRY pLCHSGeometry)
7345{
7346 int rc = VINF_SUCCESS;
7347 int rc2;
7348 bool fLockRead = false;
7349
7350 LogFlowFunc(("pDisk=%#p nImage=%u pLCHSGeometry=%#p\n",
7351 pDisk, nImage, pLCHSGeometry));
7352 do
7353 {
7354 /* sanity check */
7355 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7356 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7357
7358 /* Check arguments. */
7359 AssertMsgBreakStmt(VALID_PTR(pLCHSGeometry),
7360 ("pLCHSGeometry=%#p\n", pLCHSGeometry),
7361 rc = VERR_INVALID_PARAMETER);
7362
7363 rc2 = vdThreadStartRead(pDisk);
7364 AssertRC(rc2);
7365 fLockRead = true;
7366
7367 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7368 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7369
7370 if (pImage == pDisk->pLast)
7371 {
7372 /* Use cached information if possible. */
7373 if (pDisk->LCHSGeometry.cCylinders != 0)
7374 *pLCHSGeometry = pDisk->LCHSGeometry;
7375 else
7376 rc = VERR_VD_GEOMETRY_NOT_SET;
7377 }
7378 else
7379 rc = pImage->Backend->pfnGetLCHSGeometry(pImage->pBackendData,
7380 pLCHSGeometry);
7381 } while (0);
7382
7383 if (RT_UNLIKELY(fLockRead))
7384 {
7385 rc2 = vdThreadFinishRead(pDisk);
7386 AssertRC(rc2);
7387 }
7388
7389 LogFlowFunc((": %Rrc (LCHS=%u/%u/%u)\n", rc,
7390 pDisk->LCHSGeometry.cCylinders, pDisk->LCHSGeometry.cHeads,
7391 pDisk->LCHSGeometry.cSectors));
7392 return rc;
7393}
7394
7395/**
7396 * Store virtual disk LCHS geometry in HDD container.
7397 *
7398 * Note that in case of unrecoverable error all images in HDD container will be closed.
7399 *
7400 * @returns VBox status code.
7401 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7402 * @returns VERR_VD_GEOMETRY_NOT_SET if no geometry present in the HDD container.
7403 * @param pDisk Pointer to HDD container.
7404 * @param nImage Image number, counts from 0. 0 is always base image of container.
7405 * @param pLCHSGeometry Where to load LCHS geometry from. Not NULL.
7406 */
7407VBOXDDU_DECL(int) VDSetLCHSGeometry(PVBOXHDD pDisk, unsigned nImage,
7408 PCVDGEOMETRY pLCHSGeometry)
7409{
7410 int rc = VINF_SUCCESS;
7411 int rc2;
7412 bool fLockWrite = false;
7413
7414 LogFlowFunc(("pDisk=%#p nImage=%u pLCHSGeometry=%#p LCHS=%u/%u/%u\n",
7415 pDisk, nImage, pLCHSGeometry, pLCHSGeometry->cCylinders,
7416 pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
7417 do
7418 {
7419 /* sanity check */
7420 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7421 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7422
7423 /* Check arguments. */
7424 AssertMsgBreakStmt( VALID_PTR(pLCHSGeometry)
7425 && pLCHSGeometry->cHeads <= 255
7426 && pLCHSGeometry->cSectors <= 63,
7427 ("pLCHSGeometry=%#p LCHS=%u/%u/%u\n", pLCHSGeometry,
7428 pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads,
7429 pLCHSGeometry->cSectors),
7430 rc = VERR_INVALID_PARAMETER);
7431
7432 rc2 = vdThreadStartWrite(pDisk);
7433 AssertRC(rc2);
7434 fLockWrite = true;
7435
7436 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7437 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7438
7439 if (pImage == pDisk->pLast)
7440 {
7441 if ( pLCHSGeometry->cCylinders != pDisk->LCHSGeometry.cCylinders
7442 || pLCHSGeometry->cHeads != pDisk->LCHSGeometry.cHeads
7443 || pLCHSGeometry->cSectors != pDisk->LCHSGeometry.cSectors)
7444 {
7445 /* Only update geometry if it is changed. Avoids similar checks
7446 * in every backend. Most of the time the new geometry is set
7447 * to the previous values, so no need to go through the hassle
7448 * of updating an image which could be opened in read-only mode
7449 * right now. */
7450 rc = pImage->Backend->pfnSetLCHSGeometry(pImage->pBackendData,
7451 pLCHSGeometry);
7452
7453 /* Cache new geometry values in any case. */
7454 rc2 = pImage->Backend->pfnGetLCHSGeometry(pImage->pBackendData,
7455 &pDisk->LCHSGeometry);
7456 if (RT_FAILURE(rc2))
7457 {
7458 pDisk->LCHSGeometry.cCylinders = 0;
7459 pDisk->LCHSGeometry.cHeads = 0;
7460 pDisk->LCHSGeometry.cSectors = 0;
7461 }
7462 else
7463 {
7464 /* Make sure the CHS geometry is properly clipped. */
7465 pDisk->LCHSGeometry.cHeads = RT_MIN(pDisk->LCHSGeometry.cHeads, 255);
7466 pDisk->LCHSGeometry.cSectors = RT_MIN(pDisk->LCHSGeometry.cSectors, 63);
7467 }
7468 }
7469 }
7470 else
7471 {
7472 VDGEOMETRY LCHS;
7473 rc = pImage->Backend->pfnGetLCHSGeometry(pImage->pBackendData,
7474 &LCHS);
7475 if ( RT_FAILURE(rc)
7476 || pLCHSGeometry->cCylinders != LCHS.cCylinders
7477 || pLCHSGeometry->cHeads != LCHS.cHeads
7478 || pLCHSGeometry->cSectors != LCHS.cSectors)
7479 {
7480 /* Only update geometry if it is changed. Avoids similar checks
7481 * in every backend. Most of the time the new geometry is set
7482 * to the previous values, so no need to go through the hassle
7483 * of updating an image which could be opened in read-only mode
7484 * right now. */
7485 rc = pImage->Backend->pfnSetLCHSGeometry(pImage->pBackendData,
7486 pLCHSGeometry);
7487 }
7488 }
7489 } while (0);
7490
7491 if (RT_UNLIKELY(fLockWrite))
7492 {
7493 rc2 = vdThreadFinishWrite(pDisk);
7494 AssertRC(rc2);
7495 }
7496
7497 LogFlowFunc(("returns %Rrc\n", rc));
7498 return rc;
7499}
7500
7501/**
7502 * Get version of image in HDD container.
7503 *
7504 * @returns VBox status code.
7505 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7506 * @param pDisk Pointer to HDD container.
7507 * @param nImage Image number, counts from 0. 0 is always base image of container.
7508 * @param puVersion Where to store the image version.
7509 */
7510VBOXDDU_DECL(int) VDGetVersion(PVBOXHDD pDisk, unsigned nImage,
7511 unsigned *puVersion)
7512{
7513 int rc = VINF_SUCCESS;
7514 int rc2;
7515 bool fLockRead = false;
7516
7517 LogFlowFunc(("pDisk=%#p nImage=%u puVersion=%#p\n",
7518 pDisk, nImage, puVersion));
7519 do
7520 {
7521 /* sanity check */
7522 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7523 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7524
7525 /* Check arguments. */
7526 AssertMsgBreakStmt(VALID_PTR(puVersion),
7527 ("puVersion=%#p\n", puVersion),
7528 rc = VERR_INVALID_PARAMETER);
7529
7530 rc2 = vdThreadStartRead(pDisk);
7531 AssertRC(rc2);
7532 fLockRead = true;
7533
7534 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7535 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7536
7537 *puVersion = pImage->Backend->pfnGetVersion(pImage->pBackendData);
7538 } while (0);
7539
7540 if (RT_UNLIKELY(fLockRead))
7541 {
7542 rc2 = vdThreadFinishRead(pDisk);
7543 AssertRC(rc2);
7544 }
7545
7546 LogFlowFunc(("returns %Rrc uVersion=%#x\n", rc, *puVersion));
7547 return rc;
7548}
7549
7550/**
7551 * List the capabilities of image backend in HDD container.
7552 *
7553 * @returns VBox status code.
7554 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7555 * @param pDisk Pointer to the HDD container.
7556 * @param nImage Image number, counts from 0. 0 is always base image of container.
7557 * @param pbackendInfo Where to store the backend information.
7558 */
7559VBOXDDU_DECL(int) VDBackendInfoSingle(PVBOXHDD pDisk, unsigned nImage,
7560 PVDBACKENDINFO pBackendInfo)
7561{
7562 int rc = VINF_SUCCESS;
7563 int rc2;
7564 bool fLockRead = false;
7565
7566 LogFlowFunc(("pDisk=%#p nImage=%u pBackendInfo=%#p\n",
7567 pDisk, nImage, pBackendInfo));
7568 do
7569 {
7570 /* sanity check */
7571 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7572 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7573
7574 /* Check arguments. */
7575 AssertMsgBreakStmt(VALID_PTR(pBackendInfo),
7576 ("pBackendInfo=%#p\n", pBackendInfo),
7577 rc = VERR_INVALID_PARAMETER);
7578
7579 rc2 = vdThreadStartRead(pDisk);
7580 AssertRC(rc2);
7581 fLockRead = true;
7582
7583 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7584 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7585
7586 pBackendInfo->pszBackend = pImage->Backend->pszBackendName;
7587 pBackendInfo->uBackendCaps = pImage->Backend->uBackendCaps;
7588 pBackendInfo->paFileExtensions = pImage->Backend->paFileExtensions;
7589 pBackendInfo->paConfigInfo = pImage->Backend->paConfigInfo;
7590 } while (0);
7591
7592 if (RT_UNLIKELY(fLockRead))
7593 {
7594 rc2 = vdThreadFinishRead(pDisk);
7595 AssertRC(rc2);
7596 }
7597
7598 LogFlowFunc(("returns %Rrc\n", rc));
7599 return rc;
7600}
7601
7602/**
7603 * Get flags of image in HDD container.
7604 *
7605 * @returns VBox status code.
7606 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7607 * @param pDisk Pointer to HDD container.
7608 * @param nImage Image number, counts from 0. 0 is always base image of container.
7609 * @param puImageFlags Where to store the image flags.
7610 */
7611VBOXDDU_DECL(int) VDGetImageFlags(PVBOXHDD pDisk, unsigned nImage,
7612 unsigned *puImageFlags)
7613{
7614 int rc = VINF_SUCCESS;
7615 int rc2;
7616 bool fLockRead = false;
7617
7618 LogFlowFunc(("pDisk=%#p nImage=%u puImageFlags=%#p\n",
7619 pDisk, nImage, puImageFlags));
7620 do
7621 {
7622 /* sanity check */
7623 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7624 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7625
7626 /* Check arguments. */
7627 AssertMsgBreakStmt(VALID_PTR(puImageFlags),
7628 ("puImageFlags=%#p\n", puImageFlags),
7629 rc = VERR_INVALID_PARAMETER);
7630
7631 rc2 = vdThreadStartRead(pDisk);
7632 AssertRC(rc2);
7633 fLockRead = true;
7634
7635 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7636 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7637
7638 *puImageFlags = pImage->uImageFlags;
7639 } while (0);
7640
7641 if (RT_UNLIKELY(fLockRead))
7642 {
7643 rc2 = vdThreadFinishRead(pDisk);
7644 AssertRC(rc2);
7645 }
7646
7647 LogFlowFunc(("returns %Rrc uImageFlags=%#x\n", rc, *puImageFlags));
7648 return rc;
7649}
7650
7651/**
7652 * Get open flags of image in HDD container.
7653 *
7654 * @returns VBox status code.
7655 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7656 * @param pDisk Pointer to HDD container.
7657 * @param nImage Image number, counts from 0. 0 is always base image of container.
7658 * @param puOpenFlags Where to store the image open flags.
7659 */
7660VBOXDDU_DECL(int) VDGetOpenFlags(PVBOXHDD pDisk, unsigned nImage,
7661 unsigned *puOpenFlags)
7662{
7663 int rc = VINF_SUCCESS;
7664 int rc2;
7665 bool fLockRead = false;
7666
7667 LogFlowFunc(("pDisk=%#p nImage=%u puOpenFlags=%#p\n",
7668 pDisk, nImage, puOpenFlags));
7669 do
7670 {
7671 /* sanity check */
7672 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7673 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7674
7675 /* Check arguments. */
7676 AssertMsgBreakStmt(VALID_PTR(puOpenFlags),
7677 ("puOpenFlags=%#p\n", puOpenFlags),
7678 rc = VERR_INVALID_PARAMETER);
7679
7680 rc2 = vdThreadStartRead(pDisk);
7681 AssertRC(rc2);
7682 fLockRead = true;
7683
7684 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7685 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7686
7687 *puOpenFlags = pImage->Backend->pfnGetOpenFlags(pImage->pBackendData);
7688 } while (0);
7689
7690 if (RT_UNLIKELY(fLockRead))
7691 {
7692 rc2 = vdThreadFinishRead(pDisk);
7693 AssertRC(rc2);
7694 }
7695
7696 LogFlowFunc(("returns %Rrc uOpenFlags=%#x\n", rc, *puOpenFlags));
7697 return rc;
7698}
7699
7700/**
7701 * Set open flags of image in HDD container.
7702 * This operation may cause file locking changes and/or files being reopened.
7703 * Note that in case of unrecoverable error all images in HDD container will be closed.
7704 *
7705 * @returns VBox status code.
7706 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7707 * @param pDisk Pointer to HDD container.
7708 * @param nImage Image number, counts from 0. 0 is always base image of container.
7709 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
7710 */
7711VBOXDDU_DECL(int) VDSetOpenFlags(PVBOXHDD pDisk, unsigned nImage,
7712 unsigned uOpenFlags)
7713{
7714 int rc;
7715 int rc2;
7716 bool fLockWrite = false;
7717
7718 LogFlowFunc(("pDisk=%#p uOpenFlags=%#u\n", pDisk, uOpenFlags));
7719 do
7720 {
7721 /* sanity check */
7722 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7723 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7724
7725 /* Check arguments. */
7726 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
7727 ("uOpenFlags=%#x\n", uOpenFlags),
7728 rc = VERR_INVALID_PARAMETER);
7729
7730 rc2 = vdThreadStartWrite(pDisk);
7731 AssertRC(rc2);
7732 fLockWrite = true;
7733
7734 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7735 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7736
7737 rc = pImage->Backend->pfnSetOpenFlags(pImage->pBackendData,
7738 uOpenFlags);
7739 } while (0);
7740
7741 if (RT_UNLIKELY(fLockWrite))
7742 {
7743 rc2 = vdThreadFinishWrite(pDisk);
7744 AssertRC(rc2);
7745 }
7746
7747 LogFlowFunc(("returns %Rrc\n", rc));
7748 return rc;
7749}
7750
7751/**
7752 * Get base filename of image in HDD container. Some image formats use
7753 * other filenames as well, so don't use this for anything but informational
7754 * purposes.
7755 *
7756 * @returns VBox status code.
7757 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7758 * @returns VERR_BUFFER_OVERFLOW if pszFilename buffer too small to hold filename.
7759 * @param pDisk Pointer to HDD container.
7760 * @param nImage Image number, counts from 0. 0 is always base image of container.
7761 * @param pszFilename Where to store the image file name.
7762 * @param cbFilename Size of buffer pszFilename points to.
7763 */
7764VBOXDDU_DECL(int) VDGetFilename(PVBOXHDD pDisk, unsigned nImage,
7765 char *pszFilename, unsigned cbFilename)
7766{
7767 int rc;
7768 int rc2;
7769 bool fLockRead = false;
7770
7771 LogFlowFunc(("pDisk=%#p nImage=%u pszFilename=%#p cbFilename=%u\n",
7772 pDisk, nImage, pszFilename, cbFilename));
7773 do
7774 {
7775 /* sanity check */
7776 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7777 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7778
7779 /* Check arguments. */
7780 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
7781 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
7782 rc = VERR_INVALID_PARAMETER);
7783 AssertMsgBreakStmt(cbFilename,
7784 ("cbFilename=%u\n", cbFilename),
7785 rc = VERR_INVALID_PARAMETER);
7786
7787 rc2 = vdThreadStartRead(pDisk);
7788 AssertRC(rc2);
7789 fLockRead = true;
7790
7791 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7792 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7793
7794 size_t cb = strlen(pImage->pszFilename);
7795 if (cb <= cbFilename)
7796 {
7797 strcpy(pszFilename, pImage->pszFilename);
7798 rc = VINF_SUCCESS;
7799 }
7800 else
7801 {
7802 strncpy(pszFilename, pImage->pszFilename, cbFilename - 1);
7803 pszFilename[cbFilename - 1] = '\0';
7804 rc = VERR_BUFFER_OVERFLOW;
7805 }
7806 } while (0);
7807
7808 if (RT_UNLIKELY(fLockRead))
7809 {
7810 rc2 = vdThreadFinishRead(pDisk);
7811 AssertRC(rc2);
7812 }
7813
7814 LogFlowFunc(("returns %Rrc, pszFilename=\"%s\"\n", rc, pszFilename));
7815 return rc;
7816}
7817
7818/**
7819 * Get the comment line of image in HDD container.
7820 *
7821 * @returns VBox status code.
7822 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7823 * @returns VERR_BUFFER_OVERFLOW if pszComment buffer too small to hold comment text.
7824 * @param pDisk Pointer to HDD container.
7825 * @param nImage Image number, counts from 0. 0 is always base image of container.
7826 * @param pszComment Where to store the comment string of image. NULL is ok.
7827 * @param cbComment The size of pszComment buffer. 0 is ok.
7828 */
7829VBOXDDU_DECL(int) VDGetComment(PVBOXHDD pDisk, unsigned nImage,
7830 char *pszComment, unsigned cbComment)
7831{
7832 int rc;
7833 int rc2;
7834 bool fLockRead = false;
7835
7836 LogFlowFunc(("pDisk=%#p nImage=%u pszComment=%#p cbComment=%u\n",
7837 pDisk, nImage, pszComment, cbComment));
7838 do
7839 {
7840 /* sanity check */
7841 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7842 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7843
7844 /* Check arguments. */
7845 AssertMsgBreakStmt(VALID_PTR(pszComment),
7846 ("pszComment=%#p \"%s\"\n", pszComment, pszComment),
7847 rc = VERR_INVALID_PARAMETER);
7848 AssertMsgBreakStmt(cbComment,
7849 ("cbComment=%u\n", cbComment),
7850 rc = VERR_INVALID_PARAMETER);
7851
7852 rc2 = vdThreadStartRead(pDisk);
7853 AssertRC(rc2);
7854 fLockRead = true;
7855
7856 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7857 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7858
7859 rc = pImage->Backend->pfnGetComment(pImage->pBackendData, pszComment,
7860 cbComment);
7861 } while (0);
7862
7863 if (RT_UNLIKELY(fLockRead))
7864 {
7865 rc2 = vdThreadFinishRead(pDisk);
7866 AssertRC(rc2);
7867 }
7868
7869 LogFlowFunc(("returns %Rrc, pszComment=\"%s\"\n", rc, pszComment));
7870 return rc;
7871}
7872
7873/**
7874 * Changes the comment line of image in HDD container.
7875 *
7876 * @returns VBox status code.
7877 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7878 * @param pDisk Pointer to HDD container.
7879 * @param nImage Image number, counts from 0. 0 is always base image of container.
7880 * @param pszComment New comment string (UTF-8). NULL is allowed to reset the comment.
7881 */
7882VBOXDDU_DECL(int) VDSetComment(PVBOXHDD pDisk, unsigned nImage,
7883 const char *pszComment)
7884{
7885 int rc;
7886 int rc2;
7887 bool fLockWrite = false;
7888
7889 LogFlowFunc(("pDisk=%#p nImage=%u pszComment=%#p \"%s\"\n",
7890 pDisk, nImage, pszComment, pszComment));
7891 do
7892 {
7893 /* sanity check */
7894 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7895 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7896
7897 /* Check arguments. */
7898 AssertMsgBreakStmt(VALID_PTR(pszComment) || pszComment == NULL,
7899 ("pszComment=%#p \"%s\"\n", pszComment, pszComment),
7900 rc = VERR_INVALID_PARAMETER);
7901
7902 rc2 = vdThreadStartWrite(pDisk);
7903 AssertRC(rc2);
7904 fLockWrite = true;
7905
7906 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7907 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7908
7909 rc = pImage->Backend->pfnSetComment(pImage->pBackendData, pszComment);
7910 } while (0);
7911
7912 if (RT_UNLIKELY(fLockWrite))
7913 {
7914 rc2 = vdThreadFinishWrite(pDisk);
7915 AssertRC(rc2);
7916 }
7917
7918 LogFlowFunc(("returns %Rrc\n", rc));
7919 return rc;
7920}
7921
7922
7923/**
7924 * Get UUID of image in HDD container.
7925 *
7926 * @returns VBox status code.
7927 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7928 * @param pDisk Pointer to HDD container.
7929 * @param nImage Image number, counts from 0. 0 is always base image of container.
7930 * @param pUuid Where to store the image creation UUID.
7931 */
7932VBOXDDU_DECL(int) VDGetUuid(PVBOXHDD pDisk, unsigned nImage, PRTUUID pUuid)
7933{
7934 int rc;
7935 int rc2;
7936 bool fLockRead = false;
7937
7938 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p\n", pDisk, nImage, pUuid));
7939 do
7940 {
7941 /* sanity check */
7942 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7943 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7944
7945 /* Check arguments. */
7946 AssertMsgBreakStmt(VALID_PTR(pUuid),
7947 ("pUuid=%#p\n", pUuid),
7948 rc = VERR_INVALID_PARAMETER);
7949
7950 rc2 = vdThreadStartRead(pDisk);
7951 AssertRC(rc2);
7952 fLockRead = true;
7953
7954 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
7955 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
7956
7957 rc = pImage->Backend->pfnGetUuid(pImage->pBackendData, pUuid);
7958 } while (0);
7959
7960 if (RT_UNLIKELY(fLockRead))
7961 {
7962 rc2 = vdThreadFinishRead(pDisk);
7963 AssertRC(rc2);
7964 }
7965
7966 LogFlowFunc(("returns %Rrc, Uuid={%RTuuid}\n", rc, pUuid));
7967 return rc;
7968}
7969
7970/**
7971 * Set the image's UUID. Should not be used by normal applications.
7972 *
7973 * @returns VBox status code.
7974 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
7975 * @param pDisk Pointer to HDD container.
7976 * @param nImage Image number, counts from 0. 0 is always base image of container.
7977 * @param pUuid New UUID of the image. If NULL, a new UUID is created.
7978 */
7979VBOXDDU_DECL(int) VDSetUuid(PVBOXHDD pDisk, unsigned nImage, PCRTUUID pUuid)
7980{
7981 int rc;
7982 int rc2;
7983 bool fLockWrite = false;
7984
7985 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p {%RTuuid}\n",
7986 pDisk, nImage, pUuid, pUuid));
7987 do
7988 {
7989 /* sanity check */
7990 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
7991 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
7992
7993 AssertMsgBreakStmt(VALID_PTR(pUuid) || pUuid == NULL,
7994 ("pUuid=%#p\n", pUuid),
7995 rc = VERR_INVALID_PARAMETER);
7996
7997 rc2 = vdThreadStartWrite(pDisk);
7998 AssertRC(rc2);
7999 fLockWrite = true;
8000
8001 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
8002 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
8003
8004 RTUUID Uuid;
8005 if (!pUuid)
8006 {
8007 RTUuidCreate(&Uuid);
8008 pUuid = &Uuid;
8009 }
8010 rc = pImage->Backend->pfnSetUuid(pImage->pBackendData, pUuid);
8011 } while (0);
8012
8013 if (RT_UNLIKELY(fLockWrite))
8014 {
8015 rc2 = vdThreadFinishWrite(pDisk);
8016 AssertRC(rc2);
8017 }
8018
8019 LogFlowFunc(("returns %Rrc\n", rc));
8020 return rc;
8021}
8022
8023/**
8024 * Get last modification UUID of image in HDD container.
8025 *
8026 * @returns VBox status code.
8027 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
8028 * @param pDisk Pointer to HDD container.
8029 * @param nImage Image number, counts from 0. 0 is always base image of container.
8030 * @param pUuid Where to store the image modification UUID.
8031 */
8032VBOXDDU_DECL(int) VDGetModificationUuid(PVBOXHDD pDisk, unsigned nImage, PRTUUID pUuid)
8033{
8034 int rc = VINF_SUCCESS;
8035 int rc2;
8036 bool fLockRead = false;
8037
8038 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p\n", pDisk, nImage, pUuid));
8039 do
8040 {
8041 /* sanity check */
8042 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
8043 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
8044
8045 /* Check arguments. */
8046 AssertMsgBreakStmt(VALID_PTR(pUuid),
8047 ("pUuid=%#p\n", pUuid),
8048 rc = VERR_INVALID_PARAMETER);
8049
8050 rc2 = vdThreadStartRead(pDisk);
8051 AssertRC(rc2);
8052 fLockRead = true;
8053
8054 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
8055 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
8056
8057 rc = pImage->Backend->pfnGetModificationUuid(pImage->pBackendData,
8058 pUuid);
8059 } while (0);
8060
8061 if (RT_UNLIKELY(fLockRead))
8062 {
8063 rc2 = vdThreadFinishRead(pDisk);
8064 AssertRC(rc2);
8065 }
8066
8067 LogFlowFunc(("returns %Rrc, Uuid={%RTuuid}\n", rc, pUuid));
8068 return rc;
8069}
8070
8071/**
8072 * Set the image's last modification UUID. Should not be used by normal applications.
8073 *
8074 * @returns VBox status code.
8075 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
8076 * @param pDisk Pointer to HDD container.
8077 * @param nImage Image number, counts from 0. 0 is always base image of container.
8078 * @param pUuid New modification UUID of the image. If NULL, a new UUID is created.
8079 */
8080VBOXDDU_DECL(int) VDSetModificationUuid(PVBOXHDD pDisk, unsigned nImage, PCRTUUID pUuid)
8081{
8082 int rc;
8083 int rc2;
8084 bool fLockWrite = false;
8085
8086 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p {%RTuuid}\n",
8087 pDisk, nImage, pUuid, pUuid));
8088 do
8089 {
8090 /* sanity check */
8091 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
8092 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
8093
8094 /* Check arguments. */
8095 AssertMsgBreakStmt(VALID_PTR(pUuid) || pUuid == NULL,
8096 ("pUuid=%#p\n", pUuid),
8097 rc = VERR_INVALID_PARAMETER);
8098
8099 rc2 = vdThreadStartWrite(pDisk);
8100 AssertRC(rc2);
8101 fLockWrite = true;
8102
8103 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
8104 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
8105
8106 RTUUID Uuid;
8107 if (!pUuid)
8108 {
8109 RTUuidCreate(&Uuid);
8110 pUuid = &Uuid;
8111 }
8112 rc = pImage->Backend->pfnSetModificationUuid(pImage->pBackendData,
8113 pUuid);
8114 } while (0);
8115
8116 if (RT_UNLIKELY(fLockWrite))
8117 {
8118 rc2 = vdThreadFinishWrite(pDisk);
8119 AssertRC(rc2);
8120 }
8121
8122 LogFlowFunc(("returns %Rrc\n", rc));
8123 return rc;
8124}
8125
8126/**
8127 * Get parent UUID of image in HDD container.
8128 *
8129 * @returns VBox status code.
8130 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
8131 * @param pDisk Pointer to HDD container.
8132 * @param nImage Image number, counts from 0. 0 is always base image of container.
8133 * @param pUuid Where to store the parent image UUID.
8134 */
8135VBOXDDU_DECL(int) VDGetParentUuid(PVBOXHDD pDisk, unsigned nImage,
8136 PRTUUID pUuid)
8137{
8138 int rc = VINF_SUCCESS;
8139 int rc2;
8140 bool fLockRead = false;
8141
8142 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p\n", pDisk, nImage, pUuid));
8143 do
8144 {
8145 /* sanity check */
8146 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
8147 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
8148
8149 /* Check arguments. */
8150 AssertMsgBreakStmt(VALID_PTR(pUuid),
8151 ("pUuid=%#p\n", pUuid),
8152 rc = VERR_INVALID_PARAMETER);
8153
8154 rc2 = vdThreadStartRead(pDisk);
8155 AssertRC(rc2);
8156 fLockRead = true;
8157
8158 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
8159 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
8160
8161 rc = pImage->Backend->pfnGetParentUuid(pImage->pBackendData, pUuid);
8162 } while (0);
8163
8164 if (RT_UNLIKELY(fLockRead))
8165 {
8166 rc2 = vdThreadFinishRead(pDisk);
8167 AssertRC(rc2);
8168 }
8169
8170 LogFlowFunc(("returns %Rrc, Uuid={%RTuuid}\n", rc, pUuid));
8171 return rc;
8172}
8173
8174/**
8175 * Set the image's parent UUID. Should not be used by normal applications.
8176 *
8177 * @returns VBox status code.
8178 * @param pDisk Pointer to HDD container.
8179 * @param nImage Image number, counts from 0. 0 is always base image of container.
8180 * @param pUuid New parent UUID of the image. If NULL, a new UUID is created.
8181 */
8182VBOXDDU_DECL(int) VDSetParentUuid(PVBOXHDD pDisk, unsigned nImage,
8183 PCRTUUID pUuid)
8184{
8185 int rc;
8186 int rc2;
8187 bool fLockWrite = false;
8188
8189 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p {%RTuuid}\n",
8190 pDisk, nImage, pUuid, pUuid));
8191 do
8192 {
8193 /* sanity check */
8194 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
8195 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
8196
8197 /* Check arguments. */
8198 AssertMsgBreakStmt(VALID_PTR(pUuid) || pUuid == NULL,
8199 ("pUuid=%#p\n", pUuid),
8200 rc = VERR_INVALID_PARAMETER);
8201
8202 rc2 = vdThreadStartWrite(pDisk);
8203 AssertRC(rc2);
8204 fLockWrite = true;
8205
8206 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
8207 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
8208
8209 RTUUID Uuid;
8210 if (!pUuid)
8211 {
8212 RTUuidCreate(&Uuid);
8213 pUuid = &Uuid;
8214 }
8215 rc = pImage->Backend->pfnSetParentUuid(pImage->pBackendData, pUuid);
8216 } while (0);
8217
8218 if (RT_UNLIKELY(fLockWrite))
8219 {
8220 rc2 = vdThreadFinishWrite(pDisk);
8221 AssertRC(rc2);
8222 }
8223
8224 LogFlowFunc(("returns %Rrc\n", rc));
8225 return rc;
8226}
8227
8228
8229/**
8230 * Debug helper - dumps all opened images in HDD container into the log file.
8231 *
8232 * @param pDisk Pointer to HDD container.
8233 */
8234VBOXDDU_DECL(void) VDDumpImages(PVBOXHDD pDisk)
8235{
8236 int rc2;
8237 bool fLockRead = false;
8238
8239 do
8240 {
8241 /* sanity check */
8242 AssertPtrBreak(pDisk);
8243 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
8244
8245 if (!pDisk->pInterfaceErrorCallbacks || !VALID_PTR(pDisk->pInterfaceErrorCallbacks->pfnMessage))
8246 pDisk->pInterfaceErrorCallbacks->pfnMessage = vdLogMessage;
8247
8248 rc2 = vdThreadStartRead(pDisk);
8249 AssertRC(rc2);
8250 fLockRead = true;
8251
8252 vdMessageWrapper(pDisk, "--- Dumping VD Disk, Images=%u\n", pDisk->cImages);
8253 for (PVDIMAGE pImage = pDisk->pBase; pImage; pImage = pImage->pNext)
8254 {
8255 vdMessageWrapper(pDisk, "Dumping VD image \"%s\" (Backend=%s)\n",
8256 pImage->pszFilename, pImage->Backend->pszBackendName);
8257 pImage->Backend->pfnDump(pImage->pBackendData);
8258 }
8259 } while (0);
8260
8261 if (RT_UNLIKELY(fLockRead))
8262 {
8263 rc2 = vdThreadFinishRead(pDisk);
8264 AssertRC(rc2);
8265 }
8266}
8267
8268
8269VBOXDDU_DECL(int) VDAsyncRead(PVBOXHDD pDisk, uint64_t uOffset, size_t cbRead,
8270 PCRTSGBUF pcSgBuf,
8271 PFNVDASYNCTRANSFERCOMPLETE pfnComplete,
8272 void *pvUser1, void *pvUser2)
8273{
8274 int rc = VERR_VD_BLOCK_FREE;
8275 int rc2;
8276 bool fLockRead = false;
8277 PVDIOCTX pIoCtx = NULL;
8278
8279 LogFlowFunc(("pDisk=%#p uOffset=%llu pcSgBuf=%#p cbRead=%zu pvUser1=%#p pvUser2=%#p\n",
8280 pDisk, uOffset, pcSgBuf, cbRead, pvUser1, pvUser2));
8281
8282 do
8283 {
8284 /* sanity check */
8285 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
8286 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
8287
8288 /* Check arguments. */
8289 AssertMsgBreakStmt(cbRead,
8290 ("cbRead=%zu\n", cbRead),
8291 rc = VERR_INVALID_PARAMETER);
8292 AssertMsgBreakStmt(VALID_PTR(pcSgBuf),
8293 ("pcSgBuf=%#p\n", pcSgBuf),
8294 rc = VERR_INVALID_PARAMETER);
8295
8296 rc2 = vdThreadStartRead(pDisk);
8297 AssertRC(rc2);
8298 fLockRead = true;
8299
8300 AssertMsgBreakStmt(uOffset + cbRead <= pDisk->cbSize,
8301 ("uOffset=%llu cbRead=%zu pDisk->cbSize=%llu\n",
8302 uOffset, cbRead, pDisk->cbSize),
8303 rc = VERR_INVALID_PARAMETER);
8304 AssertPtrBreakStmt(pDisk->pLast, rc = VERR_VD_NOT_OPENED);
8305
8306 pIoCtx = vdIoCtxRootAlloc(pDisk, VDIOCTXTXDIR_READ, uOffset,
8307 cbRead, pDisk->pLast, pcSgBuf,
8308 pfnComplete, pvUser1, pvUser2,
8309 NULL, vdReadHelperAsync);
8310 if (!pIoCtx)
8311 {
8312 rc = VERR_NO_MEMORY;
8313 break;
8314 }
8315
8316 rc = vdIoCtxProcess(pIoCtx);
8317 if (rc == VINF_VD_ASYNC_IO_FINISHED)
8318 {
8319 if (ASMAtomicCmpXchgBool(&pIoCtx->fComplete, true, false))
8320 vdIoCtxFree(pDisk, pIoCtx);
8321 else
8322 rc = VERR_VD_ASYNC_IO_IN_PROGRESS; /* Let the other handler complete the request. */
8323 }
8324 else if (rc != VERR_VD_ASYNC_IO_IN_PROGRESS) /* Another error */
8325 vdIoCtxFree(pDisk, pIoCtx);
8326
8327 } while (0);
8328
8329 if (RT_UNLIKELY(fLockRead) && ( rc == VINF_VD_ASYNC_IO_FINISHED
8330 || rc != VERR_VD_ASYNC_IO_IN_PROGRESS))
8331 {
8332 rc2 = vdThreadFinishRead(pDisk);
8333 AssertRC(rc2);
8334 }
8335
8336 LogFlowFunc(("returns %Rrc\n", rc));
8337 return rc;
8338}
8339
8340
8341VBOXDDU_DECL(int) VDAsyncWrite(PVBOXHDD pDisk, uint64_t uOffset, size_t cbWrite,
8342 PCRTSGBUF pcSgBuf,
8343 PFNVDASYNCTRANSFERCOMPLETE pfnComplete,
8344 void *pvUser1, void *pvUser2)
8345{
8346 int rc;
8347 int rc2;
8348 bool fLockWrite = false;
8349 PVDIOCTX pIoCtx = NULL;
8350
8351 LogFlowFunc(("pDisk=%#p uOffset=%llu cSgBuf=%#p cbWrite=%zu pvUser1=%#p pvUser2=%#p\n",
8352 pDisk, uOffset, pcSgBuf, cbWrite, pvUser1, pvUser2));
8353 do
8354 {
8355 /* sanity check */
8356 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
8357 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
8358
8359 /* Check arguments. */
8360 AssertMsgBreakStmt(cbWrite,
8361 ("cbWrite=%zu\n", cbWrite),
8362 rc = VERR_INVALID_PARAMETER);
8363 AssertMsgBreakStmt(VALID_PTR(pcSgBuf),
8364 ("pcSgBuf=%#p\n", pcSgBuf),
8365 rc = VERR_INVALID_PARAMETER);
8366
8367 rc2 = vdThreadStartWrite(pDisk);
8368 AssertRC(rc2);
8369 fLockWrite = true;
8370
8371 AssertMsgBreakStmt(uOffset + cbWrite <= pDisk->cbSize,
8372 ("uOffset=%llu cbWrite=%zu pDisk->cbSize=%llu\n",
8373 uOffset, cbWrite, pDisk->cbSize),
8374 rc = VERR_INVALID_PARAMETER);
8375 AssertPtrBreakStmt(pDisk->pLast, rc = VERR_VD_NOT_OPENED);
8376
8377 pIoCtx = vdIoCtxRootAlloc(pDisk, VDIOCTXTXDIR_WRITE, uOffset,
8378 cbWrite, pDisk->pLast, pcSgBuf,
8379 pfnComplete, pvUser1, pvUser2,
8380 NULL, vdWriteHelperAsync);
8381 if (!pIoCtx)
8382 {
8383 rc = VERR_NO_MEMORY;
8384 break;
8385 }
8386
8387 rc = vdIoCtxProcess(pIoCtx);
8388 if (rc == VINF_VD_ASYNC_IO_FINISHED)
8389 {
8390 if (ASMAtomicCmpXchgBool(&pIoCtx->fComplete, true, false))
8391 vdIoCtxFree(pDisk, pIoCtx);
8392 else
8393 rc = VERR_VD_ASYNC_IO_IN_PROGRESS; /* Let the other handler complete the request. */
8394 }
8395 else if (rc != VERR_VD_ASYNC_IO_IN_PROGRESS) /* Another error */
8396 vdIoCtxFree(pDisk, pIoCtx);
8397 } while (0);
8398
8399 if (RT_UNLIKELY(fLockWrite) && ( rc == VINF_VD_ASYNC_IO_FINISHED
8400 || rc != VERR_VD_ASYNC_IO_IN_PROGRESS))
8401 {
8402 rc2 = vdThreadFinishWrite(pDisk);
8403 AssertRC(rc2);
8404 }
8405
8406 LogFlowFunc(("returns %Rrc\n", rc));
8407 return rc;
8408}
8409
8410
8411VBOXDDU_DECL(int) VDAsyncFlush(PVBOXHDD pDisk, PFNVDASYNCTRANSFERCOMPLETE pfnComplete,
8412 void *pvUser1, void *pvUser2)
8413{
8414 int rc;
8415 int rc2;
8416 bool fLockWrite = false;
8417 PVDIOCTX pIoCtx = NULL;
8418
8419 LogFlowFunc(("pDisk=%#p\n", pDisk));
8420
8421 do
8422 {
8423 /* sanity check */
8424 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
8425 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
8426
8427 rc2 = vdThreadStartWrite(pDisk);
8428 AssertRC(rc2);
8429 fLockWrite = true;
8430
8431 AssertPtrBreakStmt(pDisk->pLast, rc = VERR_VD_NOT_OPENED);
8432
8433 pIoCtx = vdIoCtxRootAlloc(pDisk, VDIOCTXTXDIR_FLUSH, 0,
8434 0, pDisk->pLast, NULL,
8435 pfnComplete, pvUser1, pvUser2,
8436 NULL, vdFlushHelperAsync);
8437 if (!pIoCtx)
8438 {
8439 rc = VERR_NO_MEMORY;
8440 break;
8441 }
8442
8443 rc = vdIoCtxProcess(pIoCtx);
8444 if (rc == VINF_VD_ASYNC_IO_FINISHED)
8445 {
8446 if (ASMAtomicCmpXchgBool(&pIoCtx->fComplete, true, false))
8447 vdIoCtxFree(pDisk, pIoCtx);
8448 else
8449 rc = VERR_VD_ASYNC_IO_IN_PROGRESS; /* Let the other handler complete the request. */
8450 }
8451 else if (rc != VERR_VD_ASYNC_IO_IN_PROGRESS) /* Another error */
8452 vdIoCtxFree(pDisk, pIoCtx);
8453 } while (0);
8454
8455 if (RT_UNLIKELY(fLockWrite) && ( rc == VINF_VD_ASYNC_IO_FINISHED
8456 || rc != VERR_VD_ASYNC_IO_IN_PROGRESS))
8457 {
8458 rc2 = vdThreadFinishWrite(pDisk);
8459 AssertRC(rc2);
8460 }
8461
8462 LogFlowFunc(("returns %Rrc\n", rc));
8463 return rc;
8464}
8465
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette