VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/VBoxHDD.cpp@ 22925

Last change on this file since 22925 was 21806, checked in by vboxsync, 15 years ago

Storage/VBoxHDD: resurrect the facility to dump information about disk images, and bare minimum fix for creating diff images.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 136.9 KB
Line 
1/* $Id: VBoxHDD.cpp 21806 2009-07-27 10:14:11Z vboxsync $ */
2/** @file
3 * VBoxHDD - VBox HDD Container implementation.
4 */
5
6/*
7 * Copyright (C) 2006-2008 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22/*******************************************************************************
23* Header Files *
24*******************************************************************************/
25#define LOG_GROUP LOG_GROUP_VD
26#include <VBox/VBoxHDD.h>
27#include <VBox/err.h>
28#include <VBox/sup.h>
29#include <VBox/log.h>
30
31#include <iprt/alloc.h>
32#include <iprt/assert.h>
33#include <iprt/uuid.h>
34#include <iprt/file.h>
35#include <iprt/string.h>
36#include <iprt/asm.h>
37#include <iprt/ldr.h>
38#include <iprt/dir.h>
39#include <iprt/path.h>
40#include <iprt/param.h>
41
42#include "VBoxHDD-Internal.h"
43
44
45#define VBOXHDDDISK_SIGNATURE 0x6f0e2a7d
46
47/** Buffer size used for merging images. */
48#define VD_MERGE_BUFFER_SIZE (16 * _1M)
49
50/**
51 * VBox HDD Container image descriptor.
52 */
53typedef struct VDIMAGE
54{
55 /** Link to parent image descriptor, if any. */
56 struct VDIMAGE *pPrev;
57 /** Link to child image descriptor, if any. */
58 struct VDIMAGE *pNext;
59 /** Container base filename. (UTF-8) */
60 char *pszFilename;
61 /** Data managed by the backend which keeps the actual info. */
62 void *pvBackendData;
63 /** Cached sanitized image flags. */
64 unsigned uImageFlags;
65 /** Image open flags (only those handled generically in this code and which
66 * the backends will never ever see). */
67 unsigned uOpenFlags;
68
69 /** Function pointers for the various backend methods. */
70 PCVBOXHDDBACKEND Backend;
71
72 /** Pointer to list of VD interfaces, per-image. */
73 PVDINTERFACE pVDIfsImage;
74} VDIMAGE, *PVDIMAGE;
75
76/**
77 * uModified bit flags.
78 */
79#define VD_IMAGE_MODIFIED_FLAG RT_BIT(0)
80#define VD_IMAGE_MODIFIED_FIRST RT_BIT(1)
81#define VD_IMAGE_MODIFIED_DISABLE_UUID_UPDATE RT_BIT(2)
82
83
84/**
85 * VBox HDD Container main structure, private part.
86 */
87struct VBOXHDD
88{
89 /** Structure signature (VBOXHDDDISK_SIGNATURE). */
90 uint32_t u32Signature;
91
92 /** Number of opened images. */
93 unsigned cImages;
94
95 /** Base image. */
96 PVDIMAGE pBase;
97
98 /** Last opened image in the chain.
99 * The same as pBase if only one image is used. */
100 PVDIMAGE pLast;
101
102 /** Flags representing the modification state. */
103 unsigned uModified;
104
105 /** Cached size of this disk. */
106 uint64_t cbSize;
107 /** Cached PCHS geometry for this disk. */
108 PDMMEDIAGEOMETRY PCHSGeometry;
109 /** Cached LCHS geometry for this disk. */
110 PDMMEDIAGEOMETRY LCHSGeometry;
111
112 /** Pointer to list of VD interfaces, per-disk. */
113 PVDINTERFACE pVDIfsDisk;
114 /** Pointer to the common interface structure for error reporting. */
115 PVDINTERFACE pInterfaceError;
116 /** Pointer to the error interface we use if available. */
117 PVDINTERFACEERROR pInterfaceErrorCallbacks;
118};
119
120
121/**
122 * VBox parent read descriptor, used internally for compaction.
123 */
124typedef struct VDPARENTSTATEDESC
125{
126 /** Pointer to disk descriptor. */
127 PVBOXHDD pDisk;
128 /** Pointer to image descriptor. */
129 PVDIMAGE pImage;
130} VDPARENTSTATEDESC, *PVDPARENTSTATEDESC;
131
132
133extern VBOXHDDBACKEND g_RawBackend;
134extern VBOXHDDBACKEND g_VmdkBackend;
135extern VBOXHDDBACKEND g_VDIBackend;
136extern VBOXHDDBACKEND g_VhdBackend;
137extern VBOXHDDBACKEND g_ParallelsBackend;
138#ifdef VBOX_WITH_ISCSI
139extern VBOXHDDBACKEND g_ISCSIBackend;
140#endif
141
142static unsigned g_cBackends = 0;
143static PVBOXHDDBACKEND *g_apBackends = NULL;
144static PVBOXHDDBACKEND aStaticBackends[] =
145{
146 &g_RawBackend,
147 &g_VmdkBackend,
148 &g_VDIBackend,
149 &g_VhdBackend,
150 &g_ParallelsBackend
151#ifdef VBOX_WITH_ISCSI
152 ,&g_ISCSIBackend
153#endif
154};
155
156/**
157 * internal: add several backends.
158 */
159static int vdAddBackends(PVBOXHDDBACKEND *ppBackends, unsigned cBackends)
160{
161 PVBOXHDDBACKEND *pTmp = (PVBOXHDDBACKEND*)RTMemRealloc(g_apBackends,
162 (g_cBackends + cBackends) * sizeof(PVBOXHDDBACKEND));
163 if (RT_UNLIKELY(!pTmp))
164 return VERR_NO_MEMORY;
165 g_apBackends = pTmp;
166 memcpy(&g_apBackends[g_cBackends], ppBackends, cBackends * sizeof(PVBOXHDDBACKEND));
167 g_cBackends += cBackends;
168 return VINF_SUCCESS;
169}
170
171/**
172 * internal: add single backend.
173 */
174DECLINLINE(int) vdAddBackend(PVBOXHDDBACKEND pBackend)
175{
176 return vdAddBackends(&pBackend, 1);
177}
178
179/**
180 * internal: issue error message.
181 */
182static int vdError(PVBOXHDD pDisk, int rc, RT_SRC_POS_DECL,
183 const char *pszFormat, ...)
184{
185 va_list va;
186 va_start(va, pszFormat);
187 if (pDisk->pInterfaceErrorCallbacks)
188 pDisk->pInterfaceErrorCallbacks->pfnError(pDisk->pInterfaceError->pvUser, rc, RT_SRC_POS_ARGS, pszFormat, va);
189 va_end(va);
190 return rc;
191}
192
193/**
194 * internal: find image format backend.
195 */
196static int vdFindBackend(const char *pszBackend, PCVBOXHDDBACKEND *ppBackend)
197{
198 int rc = VINF_SUCCESS;
199 PCVBOXHDDBACKEND pBackend = NULL;
200
201 if (!g_apBackends)
202 VDInit();
203
204 for (unsigned i = 0; i < g_cBackends; i++)
205 {
206 if (!RTStrICmp(pszBackend, g_apBackends[i]->pszBackendName))
207 {
208 pBackend = g_apBackends[i];
209 break;
210 }
211 }
212 *ppBackend = pBackend;
213 return rc;
214}
215
216/**
217 * internal: add image structure to the end of images list.
218 */
219static void vdAddImageToList(PVBOXHDD pDisk, PVDIMAGE pImage)
220{
221 pImage->pPrev = NULL;
222 pImage->pNext = NULL;
223
224 if (pDisk->pBase)
225 {
226 Assert(pDisk->cImages > 0);
227 pImage->pPrev = pDisk->pLast;
228 pDisk->pLast->pNext = pImage;
229 pDisk->pLast = pImage;
230 }
231 else
232 {
233 Assert(pDisk->cImages == 0);
234 pDisk->pBase = pImage;
235 pDisk->pLast = pImage;
236 }
237
238 pDisk->cImages++;
239}
240
241/**
242 * internal: remove image structure from the images list.
243 */
244static void vdRemoveImageFromList(PVBOXHDD pDisk, PVDIMAGE pImage)
245{
246 Assert(pDisk->cImages > 0);
247
248 if (pImage->pPrev)
249 pImage->pPrev->pNext = pImage->pNext;
250 else
251 pDisk->pBase = pImage->pNext;
252
253 if (pImage->pNext)
254 pImage->pNext->pPrev = pImage->pPrev;
255 else
256 pDisk->pLast = pImage->pPrev;
257
258 pImage->pPrev = NULL;
259 pImage->pNext = NULL;
260
261 pDisk->cImages--;
262}
263
264/**
265 * internal: find image by index into the images list.
266 */
267static PVDIMAGE vdGetImageByNumber(PVBOXHDD pDisk, unsigned nImage)
268{
269 PVDIMAGE pImage = pDisk->pBase;
270 if (nImage == VD_LAST_IMAGE)
271 return pDisk->pLast;
272 while (pImage && nImage)
273 {
274 pImage = pImage->pNext;
275 nImage--;
276 }
277 return pImage;
278}
279
280/**
281 * internal: read the specified amount of data in whatever blocks the backend
282 * will give us.
283 */
284static int vdReadHelper(PVBOXHDD pDisk, PVDIMAGE pImage, uint64_t uOffset,
285 void *pvBuf, size_t cbRead)
286{
287 int rc;
288 size_t cbThisRead;
289
290 /* Loop until all read. */
291 do
292 {
293 /* Search for image with allocated block. Do not attempt to read more
294 * than the previous reads marked as valid. Otherwise this would return
295 * stale data when different block sizes are used for the images. */
296 cbThisRead = cbRead;
297 rc = VERR_VD_BLOCK_FREE;
298 for (PVDIMAGE pCurrImage = pImage;
299 pCurrImage != NULL && rc == VERR_VD_BLOCK_FREE;
300 pCurrImage = pCurrImage->pPrev)
301 {
302 rc = pCurrImage->Backend->pfnRead(pCurrImage->pvBackendData,
303 uOffset, pvBuf, cbThisRead,
304 &cbThisRead);
305 }
306
307 /* No image in the chain contains the data for the block. */
308 if (rc == VERR_VD_BLOCK_FREE)
309 {
310 memset(pvBuf, '\0', cbThisRead);
311 rc = VINF_SUCCESS;
312 }
313
314 cbRead -= cbThisRead;
315 uOffset += cbThisRead;
316 pvBuf = (char *)pvBuf + cbThisRead;
317 } while (cbRead != 0 && RT_SUCCESS(rc));
318
319 return rc;
320}
321
322/**
323 * internal: parent image read wrapper for compacting.
324 */
325static int vdParentRead(void *pvUser, uint64_t uOffset, void *pvBuf,
326 size_t cbRead)
327{
328 PVDPARENTSTATEDESC pParentState = (PVDPARENTSTATEDESC)pvUser;
329 return vdReadHelper(pParentState->pDisk, pParentState->pImage, uOffset,
330 pvBuf, cbRead);
331}
332
333/**
334 * internal: mark the disk as not modified.
335 */
336static void vdResetModifiedFlag(PVBOXHDD pDisk)
337{
338 if (pDisk->uModified & VD_IMAGE_MODIFIED_FLAG)
339 {
340 /* generate new last-modified uuid */
341 if (!(pDisk->uModified & VD_IMAGE_MODIFIED_DISABLE_UUID_UPDATE))
342 {
343 RTUUID Uuid;
344
345 RTUuidCreate(&Uuid);
346 pDisk->pLast->Backend->pfnSetModificationUuid(pDisk->pLast->pvBackendData,
347 &Uuid);
348 }
349
350 pDisk->uModified &= ~VD_IMAGE_MODIFIED_FLAG;
351 }
352}
353
354/**
355 * internal: mark the disk as modified.
356 */
357static void vdSetModifiedFlag(PVBOXHDD pDisk)
358{
359 pDisk->uModified |= VD_IMAGE_MODIFIED_FLAG;
360 if (pDisk->uModified & VD_IMAGE_MODIFIED_FIRST)
361 {
362 pDisk->uModified &= ~VD_IMAGE_MODIFIED_FIRST;
363
364 /* First modify, so create a UUID and ensure it's written to disk. */
365 vdResetModifiedFlag(pDisk);
366
367 if (!(pDisk->uModified | VD_IMAGE_MODIFIED_DISABLE_UUID_UPDATE))
368 pDisk->pLast->Backend->pfnFlush(pDisk->pLast->pvBackendData);
369 }
370}
371
372/**
373 * internal: write a complete block (only used for diff images), taking the
374 * remaining data from parent images. This implementation does not optimize
375 * anything (except that it tries to read only that portions from parent
376 * images that are really needed).
377 */
378static int vdWriteHelperStandard(PVBOXHDD pDisk, PVDIMAGE pImage,
379 uint64_t uOffset, size_t cbWrite,
380 size_t cbThisWrite, size_t cbPreRead,
381 size_t cbPostRead, const void *pvBuf,
382 void *pvTmp)
383{
384 int rc = VINF_SUCCESS;
385
386 /* Read the data that goes before the write to fill the block. */
387 if (cbPreRead)
388 {
389 rc = vdReadHelper(pDisk, pImage, uOffset - cbPreRead, pvTmp,
390 cbPreRead);
391 if (RT_FAILURE(rc))
392 return rc;
393 }
394
395 /* Copy the data to the right place in the buffer. */
396 memcpy((char *)pvTmp + cbPreRead, pvBuf, cbThisWrite);
397
398 /* Read the data that goes after the write to fill the block. */
399 if (cbPostRead)
400 {
401 /* If we have data to be written, use that instead of reading
402 * data from the image. */
403 size_t cbWriteCopy;
404 if (cbWrite > cbThisWrite)
405 cbWriteCopy = RT_MIN(cbWrite - cbThisWrite, cbPostRead);
406 else
407 cbWriteCopy = 0;
408 /* Figure out how much we cannnot read from the image, because
409 * the last block to write might exceed the nominal size of the
410 * image for technical reasons. */
411 size_t cbFill;
412 if (uOffset + cbThisWrite + cbPostRead > pDisk->cbSize)
413 cbFill = uOffset + cbThisWrite + cbPostRead - pDisk->cbSize;
414 else
415 cbFill = 0;
416 /* The rest must be read from the image. */
417 size_t cbReadImage = cbPostRead - cbWriteCopy - cbFill;
418
419 /* Now assemble the remaining data. */
420 if (cbWriteCopy)
421 memcpy((char *)pvTmp + cbPreRead + cbThisWrite,
422 (char *)pvBuf + cbThisWrite, cbWriteCopy);
423 if (cbReadImage)
424 rc = vdReadHelper(pDisk, pImage,
425 uOffset + cbThisWrite + cbWriteCopy,
426 (char *)pvTmp + cbPreRead + cbThisWrite + cbWriteCopy,
427 cbReadImage);
428 if (RT_FAILURE(rc))
429 return rc;
430 /* Zero out the remainder of this block. Will never be visible, as this
431 * is beyond the limit of the image. */
432 if (cbFill)
433 memset((char *)pvTmp + cbPreRead + cbThisWrite + cbWriteCopy + cbReadImage,
434 '\0', cbFill);
435 }
436
437 /* Write the full block to the virtual disk. */
438 rc = pImage->Backend->pfnWrite(pImage->pvBackendData,
439 uOffset - cbPreRead, pvTmp,
440 cbPreRead + cbThisWrite + cbPostRead,
441 NULL, &cbPreRead, &cbPostRead, 0);
442 Assert(rc != VERR_VD_BLOCK_FREE);
443 Assert(cbPreRead == 0);
444 Assert(cbPostRead == 0);
445
446 return rc;
447}
448
449/**
450 * internal: write a complete block (only used for diff images), taking the
451 * remaining data from parent images. This implementation optimizes out writes
452 * that do not change the data relative to the state as of the parent images.
453 * All backends which support differential/growing images support this.
454 */
455static int vdWriteHelperOptimized(PVBOXHDD pDisk, PVDIMAGE pImage,
456 uint64_t uOffset, size_t cbWrite,
457 size_t cbThisWrite, size_t cbPreRead,
458 size_t cbPostRead, const void *pvBuf,
459 void *pvTmp)
460{
461 size_t cbFill = 0;
462 size_t cbWriteCopy = 0;
463 size_t cbReadImage = 0;
464 int rc;
465
466 if (cbPostRead)
467 {
468 /* Figure out how much we cannnot read from the image, because
469 * the last block to write might exceed the nominal size of the
470 * image for technical reasons. */
471 if (uOffset + cbThisWrite + cbPostRead > pDisk->cbSize)
472 cbFill = uOffset + cbThisWrite + cbPostRead - pDisk->cbSize;
473
474 /* If we have data to be written, use that instead of reading
475 * data from the image. */
476 if (cbWrite > cbThisWrite)
477 cbWriteCopy = RT_MIN(cbWrite - cbThisWrite, cbPostRead);
478
479 /* The rest must be read from the image. */
480 cbReadImage = cbPostRead - cbWriteCopy - cbFill;
481 }
482
483 /* Read the entire data of the block so that we can compare whether it will
484 * be modified by the write or not. */
485 rc = vdReadHelper(pDisk, pImage, uOffset - cbPreRead, pvTmp,
486 cbPreRead + cbThisWrite + cbPostRead - cbFill);
487 if (RT_FAILURE(rc))
488 return rc;
489
490 /* Check if the write would modify anything in this block. */
491 if ( !memcmp((char *)pvTmp + cbPreRead, pvBuf, cbThisWrite)
492 && (!cbWriteCopy || !memcmp((char *)pvTmp + cbPreRead + cbThisWrite,
493 (char *)pvBuf + cbThisWrite, cbWriteCopy)))
494 {
495 /* Block is completely unchanged, so no need to write anything. */
496 return VINF_SUCCESS;
497 }
498
499 /* Copy the data to the right place in the buffer. */
500 memcpy((char *)pvTmp + cbPreRead, pvBuf, cbThisWrite);
501
502 /* Handle the data that goes after the write to fill the block. */
503 if (cbPostRead)
504 {
505 /* Now assemble the remaining data. */
506 if (cbWriteCopy)
507 memcpy((char *)pvTmp + cbPreRead + cbThisWrite,
508 (char *)pvBuf + cbThisWrite, cbWriteCopy);
509 /* Zero out the remainder of this block. Will never be visible, as this
510 * is beyond the limit of the image. */
511 if (cbFill)
512 memset((char *)pvTmp + cbPreRead + cbThisWrite + cbWriteCopy + cbReadImage,
513 '\0', cbFill);
514 }
515
516 /* Write the full block to the virtual disk. */
517 rc = pImage->Backend->pfnWrite(pImage->pvBackendData,
518 uOffset - cbPreRead, pvTmp,
519 cbPreRead + cbThisWrite + cbPostRead,
520 NULL, &cbPreRead, &cbPostRead, 0);
521 Assert(rc != VERR_VD_BLOCK_FREE);
522 Assert(cbPreRead == 0);
523 Assert(cbPostRead == 0);
524
525 return rc;
526}
527
528/**
529 * internal: write buffer to the image, taking care of block boundaries and
530 * write optimizations.
531 */
532static int vdWriteHelper(PVBOXHDD pDisk, PVDIMAGE pImage, uint64_t uOffset,
533 const void *pvBuf, size_t cbWrite)
534{
535 int rc;
536 unsigned fWrite;
537 size_t cbThisWrite;
538 size_t cbPreRead, cbPostRead;
539
540 /* Loop until all written. */
541 do
542 {
543 /* Try to write the possibly partial block to the last opened image.
544 * This works when the block is already allocated in this image or
545 * if it is a full-block write (and allocation isn't suppressed below).
546 * For image formats which don't support zero blocks, it's beneficial
547 * to avoid unnecessarily allocating unchanged blocks. This prevents
548 * unwanted expanding of images. VMDK is an example. */
549 cbThisWrite = cbWrite;
550 fWrite = (pImage->uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME)
551 ? 0 : VD_WRITE_NO_ALLOC;
552 rc = pImage->Backend->pfnWrite(pImage->pvBackendData, uOffset, pvBuf,
553 cbThisWrite, &cbThisWrite, &cbPreRead,
554 &cbPostRead, fWrite);
555 if (rc == VERR_VD_BLOCK_FREE)
556 {
557 void *pvTmp = RTMemTmpAlloc(cbPreRead + cbThisWrite + cbPostRead);
558 AssertBreakStmt(VALID_PTR(pvTmp), rc = VERR_NO_MEMORY);
559
560 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME))
561 {
562 /* Optimized write, suppress writing to a so far unallocated
563 * block if the data is in fact not changed. */
564 rc = vdWriteHelperOptimized(pDisk, pImage, uOffset, cbWrite,
565 cbThisWrite, cbPreRead, cbPostRead,
566 pvBuf, pvTmp);
567 }
568 else
569 {
570 /* Normal write, not optimized in any way. The block will
571 * be written no matter what. This will usually (unless the
572 * backend has some further optimization enabled) cause the
573 * block to be allocated. */
574 rc = vdWriteHelperStandard(pDisk, pImage, uOffset, cbWrite,
575 cbThisWrite, cbPreRead, cbPostRead,
576 pvBuf, pvTmp);
577 }
578 RTMemTmpFree(pvTmp);
579 if (RT_FAILURE(rc))
580 break;
581 }
582
583 cbWrite -= cbThisWrite;
584 uOffset += cbThisWrite;
585 pvBuf = (char *)pvBuf + cbThisWrite;
586 } while (cbWrite != 0 && RT_SUCCESS(rc));
587
588 return rc;
589}
590
591
592/**
593 * internal: scans plugin directory and loads the backends have been found.
594 */
595static int vdLoadDynamicBackends()
596{
597 int rc = VINF_SUCCESS;
598 PRTDIR pPluginDir = NULL;
599
600 /* Enumerate plugin backends. */
601 char szPath[RTPATH_MAX];
602 rc = RTPathAppPrivateArch(szPath, sizeof(szPath));
603 if (RT_FAILURE(rc))
604 return rc;
605
606 /* To get all entries with VBoxHDD as prefix. */
607 char *pszPluginFilter;
608 rc = RTStrAPrintf(&pszPluginFilter, "%s/%s*", szPath,
609 VBOX_HDDFORMAT_PLUGIN_PREFIX);
610 if (RT_FAILURE(rc))
611 {
612 rc = VERR_NO_MEMORY;
613 return rc;
614 }
615
616 PRTDIRENTRYEX pPluginDirEntry = NULL;
617 size_t cbPluginDirEntry = sizeof(RTDIRENTRYEX);
618 /* The plugins are in the same directory as the other shared libs. */
619 rc = RTDirOpenFiltered(&pPluginDir, pszPluginFilter, RTDIRFILTER_WINNT);
620 if (RT_FAILURE(rc))
621 {
622 /* On Windows the above immediately signals that there are no
623 * files matching, while on other platforms enumerating the
624 * files below fails. Either way: no plugins. */
625 goto out;
626 }
627
628 pPluginDirEntry = (PRTDIRENTRYEX)RTMemAllocZ(sizeof(RTDIRENTRYEX));
629 if (!pPluginDirEntry)
630 {
631 rc = VERR_NO_MEMORY;
632 goto out;
633 }
634
635 while ((rc = RTDirReadEx(pPluginDir, pPluginDirEntry, &cbPluginDirEntry, RTFSOBJATTRADD_NOTHING)) != VERR_NO_MORE_FILES)
636 {
637 RTLDRMOD hPlugin = NIL_RTLDRMOD;
638 PFNVBOXHDDFORMATLOAD pfnHDDFormatLoad = NULL;
639 PVBOXHDDBACKEND pBackend = NULL;
640 char *pszPluginPath = NULL;
641
642 if (rc == VERR_BUFFER_OVERFLOW)
643 {
644 /* allocate new buffer. */
645 RTMemFree(pPluginDirEntry);
646 pPluginDirEntry = (PRTDIRENTRYEX)RTMemAllocZ(cbPluginDirEntry);
647 /* Retry. */
648 rc = RTDirReadEx(pPluginDir, pPluginDirEntry, &cbPluginDirEntry, RTFSOBJATTRADD_NOTHING);
649 if (RT_FAILURE(rc))
650 break;
651 }
652 else if (RT_FAILURE(rc))
653 break;
654
655 /* We got the new entry. */
656 if (!RTFS_IS_FILE(pPluginDirEntry->Info.Attr.fMode))
657 continue;
658
659 /* Prepend the path to the libraries. */
660 rc = RTStrAPrintf(&pszPluginPath, "%s/%s", szPath, pPluginDirEntry->szName);
661 if (RT_FAILURE(rc))
662 {
663 rc = VERR_NO_MEMORY;
664 break;
665 }
666
667 rc = SUPR3HardenedLdrLoad(pszPluginPath, &hPlugin);
668 if (RT_SUCCESS(rc))
669 {
670 rc = RTLdrGetSymbol(hPlugin, VBOX_HDDFORMAT_LOAD_NAME, (void**)&pfnHDDFormatLoad);
671 if (RT_FAILURE(rc) || !pfnHDDFormatLoad)
672 {
673 LogFunc(("error resolving the entry point %s in plugin %s, rc=%Rrc, pfnHDDFormat=%#p\n", VBOX_HDDFORMAT_LOAD_NAME, pPluginDirEntry->szName, rc, pfnHDDFormatLoad));
674 if (RT_SUCCESS(rc))
675 rc = VERR_SYMBOL_NOT_FOUND;
676 }
677
678 if (RT_SUCCESS(rc))
679 {
680 /* Get the function table. */
681 rc = pfnHDDFormatLoad(&pBackend);
682 if (RT_SUCCESS(rc) && pBackend->cbSize == sizeof(VBOXHDDBACKEND))
683 {
684 pBackend->hPlugin = hPlugin;
685 vdAddBackend(pBackend);
686 }
687 else
688 LogFunc(("ignored plugin '%s': pBackend->cbSize=%d rc=%Rrc\n", pszPluginPath, pBackend->cbSize, rc));
689 }
690 else
691 LogFunc(("ignored plugin '%s': rc=%Rrc\n", pszPluginPath, rc));
692
693 if (RT_FAILURE(rc))
694 RTLdrClose(hPlugin);
695 }
696 RTStrFree(pszPluginPath);
697 }
698out:
699 if (rc == VERR_NO_MORE_FILES)
700 rc = VINF_SUCCESS;
701 RTStrFree(pszPluginFilter);
702 if (pPluginDirEntry)
703 RTMemFree(pPluginDirEntry);
704 if (pPluginDir)
705 RTDirClose(pPluginDir);
706 return rc;
707}
708
709/**
710 * internal: send output to the log (unconditionally).
711 */
712int vdLogMessage(void *pvUser, const char *pszFormat, ...)
713{
714 NOREF(pvUser);
715 va_list args;
716 va_start(args, pszFormat);
717 RTLogPrintf(pszFormat, args);
718 va_end(args);
719 return VINF_SUCCESS;
720}
721
722
723/**
724 * Initializes HDD backends.
725 *
726 * @returns VBox status code.
727 */
728VBOXDDU_DECL(int) VDInit(void)
729{
730 int rc = vdAddBackends(aStaticBackends, RT_ELEMENTS(aStaticBackends));
731 if (RT_SUCCESS(rc))
732 rc = vdLoadDynamicBackends();
733 LogRel(("VDInit finished\n"));
734 return rc;
735}
736
737/**
738 * Destroys loaded HDD backends.
739 *
740 * @returns VBox status code.
741 */
742VBOXDDU_DECL(int) VDShutdown(void)
743{
744 PVBOXHDDBACKEND *pBackends = g_apBackends;
745 unsigned cBackends = g_cBackends;
746
747 if (!pBackends)
748 return VERR_INTERNAL_ERROR;
749
750 g_cBackends = 0;
751 g_apBackends = NULL;
752
753 for (unsigned i = 0; i < cBackends; i++)
754 if (pBackends[i]->hPlugin != NIL_RTLDRMOD)
755 RTLdrClose(pBackends[i]->hPlugin);
756
757 RTMemFree(pBackends);
758 return VINF_SUCCESS;
759}
760
761
762
763/**
764 * Lists all HDD backends and their capabilities in a caller-provided buffer.
765 *
766 * @todo this code contains memory leaks, inconsistent (and probably buggy)
767 * allocation, and it lacks documentation what the caller needs to free.
768 *
769 * @returns VBox status code.
770 * VERR_BUFFER_OVERFLOW if not enough space is passed.
771 * @param cEntriesAlloc Number of list entries available.
772 * @param pEntries Pointer to array for the entries.
773 * @param pcEntriesUsed Number of entries returned.
774 */
775VBOXDDU_DECL(int) VDBackendInfo(unsigned cEntriesAlloc, PVDBACKENDINFO pEntries,
776 unsigned *pcEntriesUsed)
777{
778 int rc = VINF_SUCCESS;
779 PRTDIR pPluginDir = NULL;
780 unsigned cEntries = 0;
781
782 LogFlowFunc(("cEntriesAlloc=%u pEntries=%#p pcEntriesUsed=%#p\n", cEntriesAlloc, pEntries, pcEntriesUsed));
783 /* Check arguments. */
784 AssertMsgReturn(cEntriesAlloc,
785 ("cEntriesAlloc=%u\n", cEntriesAlloc),
786 VERR_INVALID_PARAMETER);
787 AssertMsgReturn(VALID_PTR(pEntries),
788 ("pEntries=%#p\n", pEntries),
789 VERR_INVALID_PARAMETER);
790 AssertMsgReturn(VALID_PTR(pcEntriesUsed),
791 ("pcEntriesUsed=%#p\n", pcEntriesUsed),
792 VERR_INVALID_PARAMETER);
793 if (!g_apBackends)
794 VDInit();
795
796 if (cEntriesAlloc < g_cBackends)
797 {
798 *pcEntriesUsed = g_cBackends;
799 return VERR_BUFFER_OVERFLOW;
800 }
801
802 for (unsigned i = 0; i < g_cBackends; i++)
803 {
804 pEntries[i].pszBackend = g_apBackends[i]->pszBackendName;
805 pEntries[i].uBackendCaps = g_apBackends[i]->uBackendCaps;
806 pEntries[i].papszFileExtensions = g_apBackends[i]->papszFileExtensions;
807 pEntries[i].paConfigInfo = g_apBackends[i]->paConfigInfo;
808 pEntries[i].pfnComposeLocation = g_apBackends[i]->pfnComposeLocation;
809 pEntries[i].pfnComposeName = g_apBackends[i]->pfnComposeName;
810 }
811
812 LogFlowFunc(("returns %Rrc *pcEntriesUsed=%u\n", rc, cEntries));
813 *pcEntriesUsed = g_cBackends;
814 return rc;
815}
816
817/**
818 * Lists the capablities of a backend indentified by its name.
819 * Free all returned names with RTStrFree() when you no longer need them.
820 *
821 * @returns VBox status code.
822 * @param pszBackend The backend name.
823 * @param pEntries Pointer to an entry.
824 */
825VBOXDDU_DECL(int) VDBackendInfoOne(const char *pszBackend, PVDBACKENDINFO pEntry)
826{
827 LogFlowFunc(("pszBackend=%#p pEntry=%#p\n", pszBackend, pEntry));
828 /* Check arguments. */
829 AssertMsgReturn(VALID_PTR(pszBackend),
830 ("pszBackend=%#p\n", pszBackend),
831 VERR_INVALID_PARAMETER);
832 AssertMsgReturn(VALID_PTR(pEntry),
833 ("pEntry=%#p\n", pEntry),
834 VERR_INVALID_PARAMETER);
835 if (!g_apBackends)
836 VDInit();
837
838 /* Go through loaded backends. */
839 for (unsigned i = 0; i < g_cBackends; i++)
840 {
841 if (!RTStrICmp(pszBackend, g_apBackends[i]->pszBackendName))
842 {
843 pEntry->pszBackend = g_apBackends[i]->pszBackendName;
844 pEntry->uBackendCaps = g_apBackends[i]->uBackendCaps;
845 pEntry->papszFileExtensions = g_apBackends[i]->papszFileExtensions;
846 pEntry->paConfigInfo = g_apBackends[i]->paConfigInfo;
847 return VINF_SUCCESS;
848 }
849 }
850
851 return VERR_NOT_FOUND;
852}
853
854/**
855 * Allocates and initializes an empty HDD container.
856 * No image files are opened.
857 *
858 * @returns VBox status code.
859 * @param pVDIfsDisk Pointer to the per-disk VD interface list.
860 * @param ppDisk Where to store the reference to HDD container.
861 */
862VBOXDDU_DECL(int) VDCreate(PVDINTERFACE pVDIfsDisk, PVBOXHDD *ppDisk)
863{
864 int rc = VINF_SUCCESS;
865 PVBOXHDD pDisk = NULL;
866
867 LogFlowFunc(("pVDIfsDisk=%#p\n", pVDIfsDisk));
868 do
869 {
870 /* Check arguments. */
871 AssertMsgBreakStmt(VALID_PTR(ppDisk),
872 ("ppDisk=%#p\n", ppDisk),
873 rc = VERR_INVALID_PARAMETER);
874
875 pDisk = (PVBOXHDD)RTMemAllocZ(sizeof(VBOXHDD));
876 if (pDisk)
877 {
878 pDisk->u32Signature = VBOXHDDDISK_SIGNATURE;
879 pDisk->cImages = 0;
880 pDisk->pBase = NULL;
881 pDisk->pLast = NULL;
882 pDisk->cbSize = 0;
883 pDisk->PCHSGeometry.cCylinders = 0;
884 pDisk->PCHSGeometry.cHeads = 0;
885 pDisk->PCHSGeometry.cSectors = 0;
886 pDisk->LCHSGeometry.cCylinders = 0;
887 pDisk->LCHSGeometry.cHeads = 0;
888 pDisk->LCHSGeometry.cSectors = 0;
889 pDisk->pVDIfsDisk = pVDIfsDisk;
890 pDisk->pInterfaceError = NULL;
891 pDisk->pInterfaceErrorCallbacks = NULL;
892
893 pDisk->pInterfaceError = VDInterfaceGet(pVDIfsDisk, VDINTERFACETYPE_ERROR);
894 if (pDisk->pInterfaceError)
895 pDisk->pInterfaceErrorCallbacks = VDGetInterfaceError(pDisk->pInterfaceError);
896 *ppDisk = pDisk;
897 }
898 else
899 {
900 rc = VERR_NO_MEMORY;
901 break;
902 }
903 } while (0);
904
905 LogFlowFunc(("returns %Rrc (pDisk=%#p)\n", rc, pDisk));
906 return rc;
907}
908
909/**
910 * Destroys HDD container.
911 * If container has opened image files they will be closed.
912 *
913 * @param pDisk Pointer to HDD container.
914 */
915VBOXDDU_DECL(void) VDDestroy(PVBOXHDD pDisk)
916{
917 LogFlowFunc(("pDisk=%#p\n", pDisk));
918 do
919 {
920 /* sanity check */
921 AssertPtrBreak(pDisk);
922 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
923 VDCloseAll(pDisk);
924 RTMemFree(pDisk);
925 } while (0);
926 LogFlowFunc(("returns\n"));
927}
928
929/**
930 * Try to get the backend name which can use this image.
931 *
932 * @returns VBox status code.
933 * VINF_SUCCESS if a plugin was found.
934 * ppszFormat contains the string which can be used as backend name.
935 * VERR_NOT_SUPPORTED if no backend was found.
936 * @param pszFilename Name of the image file for which the backend is queried.
937 * @param ppszFormat Receives pointer of the UTF-8 string which contains the format name.
938 * The returned pointer must be freed using RTStrFree().
939 */
940VBOXDDU_DECL(int) VDGetFormat(const char *pszFilename, char **ppszFormat)
941{
942 int rc = VERR_NOT_SUPPORTED;
943
944 LogFlowFunc(("pszFilename=\"%s\"\n", pszFilename));
945 /* Check arguments. */
946 AssertMsgReturn(VALID_PTR(pszFilename) && *pszFilename,
947 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
948 VERR_INVALID_PARAMETER);
949 AssertMsgReturn(VALID_PTR(ppszFormat),
950 ("ppszFormat=%#p\n", ppszFormat),
951 VERR_INVALID_PARAMETER);
952
953 if (!g_apBackends)
954 VDInit();
955
956 /* Find the backend supporting this file format. */
957 for (unsigned i = 0; i < g_cBackends; i++)
958 {
959 if (g_apBackends[i]->pfnCheckIfValid)
960 {
961 rc = g_apBackends[i]->pfnCheckIfValid(pszFilename);
962 if ( RT_SUCCESS(rc)
963 /* The correct backend has been found, but there is a small
964 * incompatibility so that the file cannot be used. Stop here
965 * and signal success - the actual open will of course fail,
966 * but that will create a really sensible error message. */
967 || ( rc != VERR_VD_GEN_INVALID_HEADER
968 && rc != VERR_VD_VDI_INVALID_HEADER
969 && rc != VERR_VD_VMDK_INVALID_HEADER
970 && rc != VERR_VD_ISCSI_INVALID_HEADER
971 && rc != VERR_VD_VHD_INVALID_HEADER
972 && rc != VERR_VD_RAW_INVALID_HEADER))
973 {
974 /* Copy the name into the new string. */
975 char *pszFormat = RTStrDup(g_apBackends[i]->pszBackendName);
976 if (!pszFormat)
977 {
978 rc = VERR_NO_MEMORY;
979 break;
980 }
981 *ppszFormat = pszFormat;
982 rc = VINF_SUCCESS;
983 break;
984 }
985 rc = VERR_NOT_SUPPORTED;
986 }
987 }
988
989 LogFlowFunc(("returns %Rrc *ppszFormat=\"%s\"\n", rc, *ppszFormat));
990 return rc;
991}
992
993/**
994 * Opens an image file.
995 *
996 * The first opened image file in HDD container must have a base image type,
997 * others (next opened images) must be a differencing or undo images.
998 * Linkage is checked for differencing image to be in consistence with the previously opened image.
999 * When another differencing image is opened and the last image was opened in read/write access
1000 * mode, then the last image is reopened in read-only with deny write sharing mode. This allows
1001 * other processes to use images in read-only mode too.
1002 *
1003 * Note that the image is opened in read-only mode if a read/write open is not possible.
1004 * Use VDIsReadOnly to check open mode.
1005 *
1006 * @returns VBox status code.
1007 * @param pDisk Pointer to HDD container.
1008 * @param pszBackend Name of the image file backend to use.
1009 * @param pszFilename Name of the image file to open.
1010 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
1011 * @param pVDIfsImage Pointer to the per-image VD interface list.
1012 */
1013VBOXDDU_DECL(int) VDOpen(PVBOXHDD pDisk, const char *pszBackend,
1014 const char *pszFilename, unsigned uOpenFlags,
1015 PVDINTERFACE pVDIfsImage)
1016{
1017 int rc = VINF_SUCCESS;
1018 PVDIMAGE pImage = NULL;
1019
1020 LogFlowFunc(("pDisk=%#p pszBackend=\"%s\" pszFilename=\"%s\" uOpenFlags=%#x, pVDIfsImage=%#p\n",
1021 pDisk, pszBackend, pszFilename, uOpenFlags, pVDIfsImage));
1022 do
1023 {
1024 /* sanity check */
1025 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
1026 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
1027
1028 /* Check arguments. */
1029 AssertMsgBreakStmt(VALID_PTR(pszBackend) && *pszBackend,
1030 ("pszBackend=%#p \"%s\"\n", pszBackend, pszBackend),
1031 rc = VERR_INVALID_PARAMETER);
1032 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
1033 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
1034 rc = VERR_INVALID_PARAMETER);
1035 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
1036 ("uOpenFlags=%#x\n", uOpenFlags),
1037 rc = VERR_INVALID_PARAMETER);
1038
1039 /* Set up image descriptor. */
1040 pImage = (PVDIMAGE)RTMemAllocZ(sizeof(VDIMAGE));
1041 if (!pImage)
1042 {
1043 rc = VERR_NO_MEMORY;
1044 break;
1045 }
1046 pImage->pszFilename = RTStrDup(pszFilename);
1047 if (!pImage->pszFilename)
1048 {
1049 rc = VERR_NO_MEMORY;
1050 break;
1051 }
1052 pImage->pVDIfsImage = pVDIfsImage;
1053
1054 rc = vdFindBackend(pszBackend, &pImage->Backend);
1055 if (RT_FAILURE(rc))
1056 break;
1057 if (!pImage->Backend)
1058 {
1059 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
1060 N_("VD: unknown backend name '%s'"), pszBackend);
1061 break;
1062 }
1063
1064 pImage->uOpenFlags = uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME;
1065 rc = pImage->Backend->pfnOpen(pImage->pszFilename,
1066 uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME,
1067 pDisk->pVDIfsDisk,
1068 pImage->pVDIfsImage,
1069 &pImage->pvBackendData);
1070 /* If the open in read-write mode failed, retry in read-only mode. */
1071 if (RT_FAILURE(rc))
1072 {
1073 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY)
1074 && (rc == VERR_ACCESS_DENIED
1075 || rc == VERR_PERMISSION_DENIED
1076 || rc == VERR_WRITE_PROTECT
1077 || rc == VERR_SHARING_VIOLATION
1078 || rc == VERR_FILE_LOCK_FAILED))
1079 rc = pImage->Backend->pfnOpen(pImage->pszFilename,
1080 (uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME)
1081 | VD_OPEN_FLAGS_READONLY,
1082 pDisk->pVDIfsDisk,
1083 pImage->pVDIfsImage,
1084 &pImage->pvBackendData);
1085 if (RT_FAILURE(rc))
1086 {
1087 rc = vdError(pDisk, rc, RT_SRC_POS,
1088 N_("VD: error opening image file '%s'"), pszFilename);
1089 break;
1090 }
1091 }
1092
1093 unsigned uImageFlags;
1094 uImageFlags = pImage->Backend->pfnGetImageFlags(pImage->pvBackendData);
1095 /* Check image type. As the image itself has only partial knowledge
1096 * whether it's a base image or not, this info is derived here. The
1097 * base image can be fixed or normal, all others must be normal or
1098 * diff images. Some image formats don't distinguish between normal
1099 * and diff images, so this must be corrected here. */
1100 if (RT_FAILURE(rc))
1101 uImageFlags = VD_IMAGE_FLAGS_NONE;
1102 if ( RT_SUCCESS(rc)
1103 && !(uOpenFlags & VD_OPEN_FLAGS_INFO))
1104 {
1105 if ( pDisk->cImages == 0
1106 && (uImageFlags & VD_IMAGE_FLAGS_DIFF))
1107 {
1108 rc = VERR_VD_INVALID_TYPE;
1109 break;
1110 }
1111 else if (pDisk->cImages != 0)
1112 {
1113 if (uImageFlags & VD_IMAGE_FLAGS_FIXED)
1114 {
1115 rc = VERR_VD_INVALID_TYPE;
1116 break;
1117 }
1118 else
1119 uImageFlags |= VD_IMAGE_FLAGS_DIFF;
1120 }
1121 }
1122 pImage->uImageFlags = uImageFlags;
1123
1124 /* Force sane optimization settings. It's not worth avoiding writes
1125 * to fixed size images. The overhead would have almost no payback. */
1126 if (uImageFlags & VD_IMAGE_FLAGS_FIXED)
1127 pImage->uOpenFlags |= VD_OPEN_FLAGS_HONOR_SAME;
1128
1129 /** @todo optionally check UUIDs */
1130
1131 int rc2;
1132
1133 /* Cache disk information. */
1134 pDisk->cbSize = pImage->Backend->pfnGetSize(pImage->pvBackendData);
1135
1136 /* Cache PCHS geometry. */
1137 rc2 = pImage->Backend->pfnGetPCHSGeometry(pImage->pvBackendData,
1138 &pDisk->PCHSGeometry);
1139 if (RT_FAILURE(rc2))
1140 {
1141 pDisk->PCHSGeometry.cCylinders = 0;
1142 pDisk->PCHSGeometry.cHeads = 0;
1143 pDisk->PCHSGeometry.cSectors = 0;
1144 }
1145 else
1146 {
1147 /* Make sure the PCHS geometry is properly clipped. */
1148 pDisk->PCHSGeometry.cCylinders = RT_MIN(pDisk->PCHSGeometry.cCylinders, 16383);
1149 pDisk->PCHSGeometry.cHeads = RT_MIN(pDisk->PCHSGeometry.cHeads, 16);
1150 pDisk->PCHSGeometry.cSectors = RT_MIN(pDisk->PCHSGeometry.cSectors, 63);
1151 }
1152
1153 /* Cache LCHS geometry. */
1154 rc2 = pImage->Backend->pfnGetLCHSGeometry(pImage->pvBackendData,
1155 &pDisk->LCHSGeometry);
1156 if (RT_FAILURE(rc2))
1157 {
1158 pDisk->LCHSGeometry.cCylinders = 0;
1159 pDisk->LCHSGeometry.cHeads = 0;
1160 pDisk->LCHSGeometry.cSectors = 0;
1161 }
1162 else
1163 {
1164 /* Make sure the LCHS geometry is properly clipped. */
1165 pDisk->LCHSGeometry.cHeads = RT_MIN(pDisk->LCHSGeometry.cHeads, 255);
1166 pDisk->LCHSGeometry.cSectors = RT_MIN(pDisk->LCHSGeometry.cSectors, 63);
1167 }
1168
1169 if (pDisk->cImages != 0)
1170 {
1171 /* Switch previous image to read-only mode. */
1172 unsigned uOpenFlagsPrevImg;
1173 uOpenFlagsPrevImg = pDisk->pLast->Backend->pfnGetOpenFlags(pDisk->pLast->pvBackendData);
1174 if (!(uOpenFlagsPrevImg & VD_OPEN_FLAGS_READONLY))
1175 {
1176 uOpenFlagsPrevImg |= VD_OPEN_FLAGS_READONLY;
1177 rc = pDisk->pLast->Backend->pfnSetOpenFlags(pDisk->pLast->pvBackendData, uOpenFlagsPrevImg);
1178 }
1179 }
1180
1181 if (RT_SUCCESS(rc))
1182 {
1183 /* Image successfully opened, make it the last image. */
1184 vdAddImageToList(pDisk, pImage);
1185 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY))
1186 pDisk->uModified = VD_IMAGE_MODIFIED_FIRST;
1187 }
1188 else
1189 {
1190 /* Error detected, but image opened. Close image. */
1191 int rc2;
1192 rc2 = pImage->Backend->pfnClose(pImage->pvBackendData, false);
1193 AssertRC(rc2);
1194 pImage->pvBackendData = NULL;
1195 }
1196 } while (0);
1197
1198 if (RT_FAILURE(rc))
1199 {
1200 if (pImage)
1201 {
1202 if (pImage->pszFilename)
1203 RTStrFree(pImage->pszFilename);
1204 RTMemFree(pImage);
1205 }
1206 }
1207
1208 LogFlowFunc(("returns %Rrc\n", rc));
1209 return rc;
1210}
1211
1212/**
1213 * Creates and opens a new base image file.
1214 *
1215 * @returns VBox status code.
1216 * @param pDisk Pointer to HDD container.
1217 * @param pszBackend Name of the image file backend to use.
1218 * @param pszFilename Name of the image file to create.
1219 * @param cbSize Image size in bytes.
1220 * @param uImageFlags Flags specifying special image features.
1221 * @param pszComment Pointer to image comment. NULL is ok.
1222 * @param pPCHSGeometry Pointer to physical disk geometry <= (16383,16,63). Not NULL.
1223 * @param pLCHSGeometry Pointer to logical disk geometry <= (x,255,63). Not NULL.
1224 * @param pUuid New UUID of the image. If NULL, a new UUID is created.
1225 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
1226 * @param pVDIfsImage Pointer to the per-image VD interface list.
1227 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
1228 */
1229VBOXDDU_DECL(int) VDCreateBase(PVBOXHDD pDisk, const char *pszBackend,
1230 const char *pszFilename, uint64_t cbSize,
1231 unsigned uImageFlags, const char *pszComment,
1232 PCPDMMEDIAGEOMETRY pPCHSGeometry,
1233 PCPDMMEDIAGEOMETRY pLCHSGeometry,
1234 PCRTUUID pUuid, unsigned uOpenFlags,
1235 PVDINTERFACE pVDIfsImage,
1236 PVDINTERFACE pVDIfsOperation)
1237{
1238 int rc = VINF_SUCCESS;
1239 PVDIMAGE pImage = NULL;
1240 RTUUID uuid;
1241
1242 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",
1243 pDisk, pszBackend, pszFilename, cbSize, uImageFlags, pszComment,
1244 pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads,
1245 pPCHSGeometry->cSectors, pLCHSGeometry->cCylinders,
1246 pLCHSGeometry->cHeads, pLCHSGeometry->cSectors, pUuid,
1247 uOpenFlags, pVDIfsImage, pVDIfsOperation));
1248
1249 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
1250 VDINTERFACETYPE_PROGRESS);
1251 PVDINTERFACEPROGRESS pCbProgress = NULL;
1252 if (pIfProgress)
1253 pCbProgress = VDGetInterfaceProgress(pIfProgress);
1254
1255 do
1256 {
1257 /* sanity check */
1258 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
1259 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
1260
1261 /* Check arguments. */
1262 AssertMsgBreakStmt(VALID_PTR(pszBackend) && *pszBackend,
1263 ("pszBackend=%#p \"%s\"\n", pszBackend, pszBackend),
1264 rc = VERR_INVALID_PARAMETER);
1265 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
1266 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
1267 rc = VERR_INVALID_PARAMETER);
1268 AssertMsgBreakStmt(cbSize,
1269 ("cbSize=%llu\n", cbSize),
1270 rc = VERR_INVALID_PARAMETER);
1271 AssertMsgBreakStmt( ((uImageFlags & ~VD_IMAGE_FLAGS_MASK) == 0)
1272 || ((uImageFlags & (VD_IMAGE_FLAGS_FIXED | VD_IMAGE_FLAGS_DIFF)) != VD_IMAGE_FLAGS_FIXED),
1273 ("uImageFlags=%#x\n", uImageFlags),
1274 rc = VERR_INVALID_PARAMETER);
1275 /* The PCHS geometry fields may be 0 to leave it for later. */
1276 AssertMsgBreakStmt( VALID_PTR(pPCHSGeometry)
1277 && pPCHSGeometry->cHeads <= 16
1278 && pPCHSGeometry->cSectors <= 63,
1279 ("pPCHSGeometry=%#p PCHS=%u/%u/%u\n", pPCHSGeometry,
1280 pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads,
1281 pPCHSGeometry->cSectors),
1282 rc = VERR_INVALID_PARAMETER);
1283 /* The LCHS geometry fields may be 0 to leave it to later autodetection. */
1284 AssertMsgBreakStmt( VALID_PTR(pLCHSGeometry)
1285 && pLCHSGeometry->cHeads <= 255
1286 && pLCHSGeometry->cSectors <= 63,
1287 ("pLCHSGeometry=%#p LCHS=%u/%u/%u\n", pLCHSGeometry,
1288 pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads,
1289 pLCHSGeometry->cSectors),
1290 rc = VERR_INVALID_PARAMETER);
1291 /* The UUID may be NULL. */
1292 AssertMsgBreakStmt(pUuid == NULL || VALID_PTR(pUuid),
1293 ("pUuid=%#p UUID=%RTuuid\n", pUuid, pUuid),
1294 rc = VERR_INVALID_PARAMETER);
1295 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
1296 ("uOpenFlags=%#x\n", uOpenFlags),
1297 rc = VERR_INVALID_PARAMETER);
1298
1299 /* Check state. */
1300 AssertMsgBreakStmt(pDisk->cImages == 0,
1301 ("Create base image cannot be done with other images open\n"),
1302 rc = VERR_VD_INVALID_STATE);
1303
1304 /* Set up image descriptor. */
1305 pImage = (PVDIMAGE)RTMemAllocZ(sizeof(VDIMAGE));
1306 if (!pImage)
1307 {
1308 rc = VERR_NO_MEMORY;
1309 break;
1310 }
1311 pImage->pszFilename = RTStrDup(pszFilename);
1312 if (!pImage->pszFilename)
1313 {
1314 rc = VERR_NO_MEMORY;
1315 break;
1316 }
1317 pImage->pVDIfsImage = pVDIfsImage;
1318
1319 rc = vdFindBackend(pszBackend, &pImage->Backend);
1320 if (RT_FAILURE(rc))
1321 break;
1322 if (!pImage->Backend)
1323 {
1324 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
1325 N_("VD: unknown backend name '%s'"), pszBackend);
1326 break;
1327 }
1328
1329 /* Create UUID if the caller didn't specify one. */
1330 if (!pUuid)
1331 {
1332 rc = RTUuidCreate(&uuid);
1333 if (RT_FAILURE(rc))
1334 {
1335 rc = vdError(pDisk, rc, RT_SRC_POS,
1336 N_("VD: cannot generate UUID for image '%s'"),
1337 pszFilename);
1338 break;
1339 }
1340 pUuid = &uuid;
1341 }
1342
1343 pImage->uOpenFlags = uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME;
1344 uImageFlags &= ~VD_IMAGE_FLAGS_DIFF;
1345 rc = pImage->Backend->pfnCreate(pImage->pszFilename, cbSize,
1346 uImageFlags, pszComment, pPCHSGeometry,
1347 pLCHSGeometry, pUuid,
1348 uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME,
1349 0, 99,
1350 pDisk->pVDIfsDisk,
1351 pImage->pVDIfsImage,
1352 pVDIfsOperation,
1353 &pImage->pvBackendData);
1354
1355 if (RT_SUCCESS(rc))
1356 {
1357 pImage->uImageFlags = uImageFlags;
1358
1359 /* Force sane optimization settings. It's not worth avoiding writes
1360 * to fixed size images. The overhead would have almost no payback. */
1361 if (uImageFlags & VD_IMAGE_FLAGS_FIXED)
1362 pImage->uOpenFlags |= VD_OPEN_FLAGS_HONOR_SAME;
1363
1364 /** @todo optionally check UUIDs */
1365
1366 int rc2;
1367
1368 /* Cache disk information. */
1369 pDisk->cbSize = pImage->Backend->pfnGetSize(pImage->pvBackendData);
1370
1371 /* Cache PCHS geometry. */
1372 rc2 = pImage->Backend->pfnGetPCHSGeometry(pImage->pvBackendData,
1373 &pDisk->PCHSGeometry);
1374 if (RT_FAILURE(rc2))
1375 {
1376 pDisk->PCHSGeometry.cCylinders = 0;
1377 pDisk->PCHSGeometry.cHeads = 0;
1378 pDisk->PCHSGeometry.cSectors = 0;
1379 }
1380 else
1381 {
1382 /* Make sure the CHS geometry is properly clipped. */
1383 pDisk->PCHSGeometry.cCylinders = RT_MIN(pDisk->PCHSGeometry.cCylinders, 16383);
1384 pDisk->PCHSGeometry.cHeads = RT_MIN(pDisk->PCHSGeometry.cHeads, 16);
1385 pDisk->PCHSGeometry.cSectors = RT_MIN(pDisk->PCHSGeometry.cSectors, 63);
1386 }
1387
1388 /* Cache LCHS geometry. */
1389 rc2 = pImage->Backend->pfnGetLCHSGeometry(pImage->pvBackendData,
1390 &pDisk->LCHSGeometry);
1391 if (RT_FAILURE(rc2))
1392 {
1393 pDisk->LCHSGeometry.cCylinders = 0;
1394 pDisk->LCHSGeometry.cHeads = 0;
1395 pDisk->LCHSGeometry.cSectors = 0;
1396 }
1397 else
1398 {
1399 /* Make sure the CHS geometry is properly clipped. */
1400 pDisk->LCHSGeometry.cHeads = RT_MIN(pDisk->LCHSGeometry.cHeads, 255);
1401 pDisk->LCHSGeometry.cSectors = RT_MIN(pDisk->LCHSGeometry.cSectors, 63);
1402 }
1403 }
1404
1405 if (RT_SUCCESS(rc))
1406 {
1407 /* Image successfully opened, make it the last image. */
1408 vdAddImageToList(pDisk, pImage);
1409 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY))
1410 pDisk->uModified = VD_IMAGE_MODIFIED_FIRST;
1411 }
1412 else
1413 {
1414 /* Error detected, but image opened. Close and delete image. */
1415 int rc2;
1416 rc2 = pImage->Backend->pfnClose(pImage->pvBackendData, true);
1417 AssertRC(rc2);
1418 pImage->pvBackendData = NULL;
1419 }
1420 } while (0);
1421
1422 if (RT_FAILURE(rc))
1423 {
1424 if (pImage)
1425 {
1426 if (pImage->pszFilename)
1427 RTStrFree(pImage->pszFilename);
1428 RTMemFree(pImage);
1429 }
1430 }
1431
1432 if (RT_SUCCESS(rc) && pCbProgress && pCbProgress->pfnProgress)
1433 pCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */, 100,
1434 pIfProgress->pvUser);
1435
1436 LogFlowFunc(("returns %Rrc\n", rc));
1437 return rc;
1438}
1439
1440/**
1441 * Creates and opens a new differencing image file in HDD container.
1442 * See comments for VDOpen function about differencing images.
1443 *
1444 * @returns VBox status code.
1445 * @param pDisk Pointer to HDD container.
1446 * @param pszBackend Name of the image file backend to use.
1447 * @param pszFilename Name of the differencing image file to create.
1448 * @param uImageFlags Flags specifying special image features.
1449 * @param pszComment Pointer to image comment. NULL is ok.
1450 * @param pUuid New UUID of the image. If NULL, a new UUID is created.
1451 * @param pParentUuid New parent UUID of the image. If NULL, the UUID is queried automatically.
1452 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
1453 * @param pVDIfsImage Pointer to the per-image VD interface list.
1454 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
1455 */
1456VBOXDDU_DECL(int) VDCreateDiff(PVBOXHDD pDisk, const char *pszBackend,
1457 const char *pszFilename, unsigned uImageFlags,
1458 const char *pszComment, PCRTUUID pUuid,
1459 PCRTUUID pParentUuid, unsigned uOpenFlags,
1460 PVDINTERFACE pVDIfsImage,
1461 PVDINTERFACE pVDIfsOperation)
1462{
1463 int rc = VINF_SUCCESS;
1464 PVDIMAGE pImage = NULL;
1465 RTUUID uuid;
1466
1467 LogFlowFunc(("pDisk=%#p pszBackend=\"%s\" pszFilename=\"%s\" uImageFlags=%#x pszComment=\"%s\" Uuid=%RTuuid ParentUuid=%RTuuid uOpenFlags=%#x pVDIfsImage=%#p pVDIfsOperation=%#p\n",
1468 pDisk, pszBackend, pszFilename, uImageFlags, pszComment, pUuid, pParentUuid, uOpenFlags,
1469 pVDIfsImage, pVDIfsOperation));
1470
1471 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
1472 VDINTERFACETYPE_PROGRESS);
1473 PVDINTERFACEPROGRESS pCbProgress = NULL;
1474 if (pIfProgress)
1475 pCbProgress = VDGetInterfaceProgress(pIfProgress);
1476
1477 do
1478 {
1479 /* sanity check */
1480 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
1481 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
1482
1483 /* Check arguments. */
1484 AssertMsgBreakStmt(VALID_PTR(pszBackend) && *pszBackend,
1485 ("pszBackend=%#p \"%s\"\n", pszBackend, pszBackend),
1486 rc = VERR_INVALID_PARAMETER);
1487 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
1488 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
1489 rc = VERR_INVALID_PARAMETER);
1490 AssertMsgBreakStmt((uImageFlags & ~VD_IMAGE_FLAGS_MASK) == 0,
1491 ("uImageFlags=%#x\n", uImageFlags),
1492 rc = VERR_INVALID_PARAMETER);
1493 /* The UUID may be NULL. */
1494 AssertMsgBreakStmt(pUuid == NULL || VALID_PTR(pUuid),
1495 ("pUuid=%#p UUID=%RTuuid\n", pUuid, pUuid),
1496 rc = VERR_INVALID_PARAMETER);
1497 /* The parent UUID may be NULL. */
1498 AssertMsgBreakStmt(pParentUuid == NULL || VALID_PTR(pParentUuid),
1499 ("pParentUuid=%#p ParentUUID=%RTuuid\n", pParentUuid, pParentUuid),
1500 rc = VERR_INVALID_PARAMETER);
1501 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
1502 ("uOpenFlags=%#x\n", uOpenFlags),
1503 rc = VERR_INVALID_PARAMETER);
1504
1505 /* Check state. */
1506 AssertMsgBreakStmt(pDisk->cImages != 0,
1507 ("Create diff image cannot be done without other images open\n"),
1508 rc = VERR_VD_INVALID_STATE);
1509
1510 /* Set up image descriptor. */
1511 pImage = (PVDIMAGE)RTMemAllocZ(sizeof(VDIMAGE));
1512 if (!pImage)
1513 {
1514 rc = VERR_NO_MEMORY;
1515 break;
1516 }
1517 pImage->pszFilename = RTStrDup(pszFilename);
1518 if (!pImage->pszFilename)
1519 {
1520 rc = VERR_NO_MEMORY;
1521 break;
1522 }
1523
1524 rc = vdFindBackend(pszBackend, &pImage->Backend);
1525 if (RT_FAILURE(rc))
1526 break;
1527 if (!pImage->Backend)
1528 {
1529 rc = vdError(pDisk, VERR_INVALID_PARAMETER, RT_SRC_POS,
1530 N_("VD: unknown backend name '%s'"), pszBackend);
1531 break;
1532 }
1533
1534 /* Create UUID if the caller didn't specify one. */
1535 if (!pUuid)
1536 {
1537 rc = RTUuidCreate(&uuid);
1538 if (RT_FAILURE(rc))
1539 {
1540 rc = vdError(pDisk, rc, RT_SRC_POS,
1541 N_("VD: cannot generate UUID for image '%s'"),
1542 pszFilename);
1543 break;
1544 }
1545 pUuid = &uuid;
1546 }
1547
1548 pImage->uOpenFlags = uOpenFlags & VD_OPEN_FLAGS_HONOR_SAME;
1549 uImageFlags |= VD_IMAGE_FLAGS_DIFF;
1550 rc = pImage->Backend->pfnCreate(pImage->pszFilename, pDisk->cbSize,
1551 uImageFlags | VD_IMAGE_FLAGS_DIFF,
1552 pszComment, &pDisk->PCHSGeometry,
1553 &pDisk->LCHSGeometry, pUuid,
1554 uOpenFlags & ~VD_OPEN_FLAGS_HONOR_SAME,
1555 0, 99,
1556 pDisk->pVDIfsDisk,
1557 pImage->pVDIfsImage,
1558 pVDIfsOperation,
1559 &pImage->pvBackendData);
1560
1561 if (RT_SUCCESS(rc) && pDisk->cImages != 0)
1562 {
1563 pImage->uImageFlags = uImageFlags;
1564
1565 /* Switch previous image to read-only mode. */
1566 unsigned uOpenFlagsPrevImg;
1567 uOpenFlagsPrevImg = pDisk->pLast->Backend->pfnGetOpenFlags(pDisk->pLast->pvBackendData);
1568 if (!(uOpenFlagsPrevImg & VD_OPEN_FLAGS_READONLY))
1569 {
1570 uOpenFlagsPrevImg |= VD_OPEN_FLAGS_READONLY;
1571 rc = pDisk->pLast->Backend->pfnSetOpenFlags(pDisk->pLast->pvBackendData, uOpenFlagsPrevImg);
1572 }
1573 }
1574
1575 if (RT_SUCCESS(rc))
1576 {
1577 RTUUID Uuid;
1578 RTTIMESPEC ts;
1579 int rc2;
1580
1581 if (pParentUuid && !RTUuidIsNull(pParentUuid))
1582 {
1583 Uuid = *pParentUuid;
1584 pImage->Backend->pfnSetParentUuid(pImage->pvBackendData, &Uuid);
1585 }
1586 else
1587 {
1588 rc2 = pDisk->pLast->Backend->pfnGetUuid(pDisk->pLast->pvBackendData,
1589 &Uuid);
1590 if (RT_SUCCESS(rc2))
1591 pImage->Backend->pfnSetParentUuid(pImage->pvBackendData, &Uuid);
1592 }
1593 rc2 = pDisk->pLast->Backend->pfnGetModificationUuid(pDisk->pLast->pvBackendData,
1594 &Uuid);
1595 if (RT_SUCCESS(rc2))
1596 pImage->Backend->pfnSetParentModificationUuid(pImage->pvBackendData,
1597 &Uuid);
1598 rc2 = pDisk->pLast->Backend->pfnGetTimeStamp(pDisk->pLast->pvBackendData,
1599 &ts);
1600 if (RT_SUCCESS(rc2))
1601 pImage->Backend->pfnSetParentTimeStamp(pImage->pvBackendData, &ts);
1602
1603 rc2 = pImage->Backend->pfnSetParentFilename(pImage->pvBackendData, pDisk->pLast->pszFilename);
1604 }
1605
1606 if (RT_SUCCESS(rc))
1607 {
1608 /** @todo optionally check UUIDs */
1609 }
1610
1611 if (RT_SUCCESS(rc))
1612 {
1613 /* Image successfully opened, make it the last image. */
1614 vdAddImageToList(pDisk, pImage);
1615 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY))
1616 pDisk->uModified = VD_IMAGE_MODIFIED_FIRST;
1617 }
1618 else
1619 {
1620 /* Error detected, but image opened. Close and delete image. */
1621 int rc2;
1622 rc2 = pImage->Backend->pfnClose(pImage->pvBackendData, true);
1623 AssertRC(rc2);
1624 pImage->pvBackendData = NULL;
1625 }
1626 } while (0);
1627
1628 if (RT_FAILURE(rc))
1629 {
1630 if (pImage)
1631 {
1632 if (pImage->pszFilename)
1633 RTStrFree(pImage->pszFilename);
1634 RTMemFree(pImage);
1635 }
1636 }
1637
1638 if (RT_SUCCESS(rc) && pCbProgress && pCbProgress->pfnProgress)
1639 pCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */, 100,
1640 pIfProgress->pvUser);
1641
1642 LogFlowFunc(("returns %Rrc\n", rc));
1643 return rc;
1644}
1645
1646/**
1647 * Merges two images (not necessarily with direct parent/child relationship).
1648 * As a side effect the source image and potentially the other images which
1649 * are also merged to the destination are deleted from both the disk and the
1650 * images in the HDD container.
1651 *
1652 * @returns VBox status code.
1653 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
1654 * @param pDisk Pointer to HDD container.
1655 * @param nImageFrom Name of the image file to merge from.
1656 * @param nImageTo Name of the image file to merge to.
1657 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
1658 */
1659VBOXDDU_DECL(int) VDMerge(PVBOXHDD pDisk, unsigned nImageFrom,
1660 unsigned nImageTo, PVDINTERFACE pVDIfsOperation)
1661{
1662 int rc = VINF_SUCCESS;
1663 void *pvBuf = NULL;
1664
1665 LogFlowFunc(("pDisk=%#p nImageFrom=%u nImageTo=%u pVDIfsOperation=%#p\n",
1666 pDisk, nImageFrom, nImageTo, pVDIfsOperation));
1667
1668 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
1669 VDINTERFACETYPE_PROGRESS);
1670 PVDINTERFACEPROGRESS pCbProgress = NULL;
1671 if (pIfProgress)
1672 pCbProgress = VDGetInterfaceProgress(pIfProgress);
1673
1674 do
1675 {
1676 /* sanity check */
1677 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
1678 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
1679
1680 PVDIMAGE pImageFrom = vdGetImageByNumber(pDisk, nImageFrom);
1681 PVDIMAGE pImageTo = vdGetImageByNumber(pDisk, nImageTo);
1682 if (!pImageFrom || !pImageTo)
1683 {
1684 rc = VERR_VD_IMAGE_NOT_FOUND;
1685 break;
1686 }
1687 AssertBreakStmt(pImageFrom != pImageTo, rc = VERR_INVALID_PARAMETER);
1688
1689 /* Make sure destination image is writable. */
1690 unsigned uOpenFlags = pImageTo->Backend->pfnGetOpenFlags(pImageTo->pvBackendData);
1691 if (uOpenFlags & VD_OPEN_FLAGS_READONLY)
1692 {
1693 uOpenFlags &= ~VD_OPEN_FLAGS_READONLY;
1694 rc = pImageTo->Backend->pfnSetOpenFlags(pImageTo->pvBackendData,
1695 uOpenFlags);
1696 if (RT_FAILURE(rc))
1697 break;
1698 }
1699
1700 /* Get size of destination image. */
1701 uint64_t cbSize = pImageTo->Backend->pfnGetSize(pImageTo->pvBackendData);
1702
1703 /* Allocate tmp buffer. */
1704 pvBuf = RTMemTmpAlloc(VD_MERGE_BUFFER_SIZE);
1705 if (!pvBuf)
1706 {
1707 rc = VERR_NO_MEMORY;
1708 break;
1709 }
1710
1711 /* Merging is done directly on the images itself. This potentially
1712 * causes trouble if the disk is full in the middle of operation. */
1713 /** @todo write alternative implementation which works with temporary
1714 * images (which is safer, but requires even more space). Also has the
1715 * drawback that merging into a raw disk parent simply isn't possible
1716 * this way (but in that case disk full isn't really a problem). */
1717 if (nImageFrom < nImageTo)
1718 {
1719 /* Merge parent state into child. This means writing all not
1720 * allocated blocks in the destination image which are allocated in
1721 * the images to be merged. */
1722 uint64_t uOffset = 0;
1723 uint64_t cbRemaining = cbSize;
1724 do
1725 {
1726 size_t cbThisRead = RT_MIN(VD_MERGE_BUFFER_SIZE, cbRemaining);
1727 rc = pImageTo->Backend->pfnRead(pImageTo->pvBackendData,
1728 uOffset, pvBuf, cbThisRead,
1729 &cbThisRead);
1730 if (rc == VERR_VD_BLOCK_FREE)
1731 {
1732 /* Search for image with allocated block. Do not attempt to
1733 * read more than the previous reads marked as valid.
1734 * Otherwise this would return stale data when different
1735 * block sizes are used for the images. */
1736 for (PVDIMAGE pCurrImage = pImageTo->pPrev;
1737 pCurrImage != NULL && pCurrImage != pImageFrom->pPrev && rc == VERR_VD_BLOCK_FREE;
1738 pCurrImage = pCurrImage->pPrev)
1739 {
1740 rc = pCurrImage->Backend->pfnRead(pCurrImage->pvBackendData,
1741 uOffset, pvBuf,
1742 cbThisRead,
1743 &cbThisRead);
1744 }
1745
1746 if (rc != VERR_VD_BLOCK_FREE)
1747 {
1748 if (RT_FAILURE(rc))
1749 break;
1750 rc = vdWriteHelper(pDisk, pImageTo, uOffset, pvBuf,
1751 cbThisRead);
1752 if (RT_FAILURE(rc))
1753 break;
1754 }
1755 else
1756 rc = VINF_SUCCESS;
1757 }
1758 else if (RT_FAILURE(rc))
1759 break;
1760
1761 uOffset += cbThisRead;
1762 cbRemaining -= cbThisRead;
1763
1764 if (pCbProgress && pCbProgress->pfnProgress)
1765 {
1766 rc = pCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */,
1767 uOffset * 99 / cbSize,
1768 pIfProgress->pvUser);
1769 if (RT_FAILURE(rc))
1770 break;
1771 }
1772 } while (uOffset < cbSize);
1773 }
1774 else
1775 {
1776 /* Merge child state into parent. This means writing all blocks
1777 * which are allocated in the image up to the source image to the
1778 * destination image. */
1779 uint64_t uOffset = 0;
1780 uint64_t cbRemaining = cbSize;
1781 do
1782 {
1783 size_t cbThisRead = RT_MIN(VD_MERGE_BUFFER_SIZE, cbRemaining);
1784 rc = VERR_VD_BLOCK_FREE;
1785 /* Search for image with allocated block. Do not attempt to
1786 * read more than the previous reads marked as valid. Otherwise
1787 * this would return stale data when different block sizes are
1788 * used for the images. */
1789 for (PVDIMAGE pCurrImage = pImageFrom;
1790 pCurrImage != NULL && pCurrImage != pImageTo && rc == VERR_VD_BLOCK_FREE;
1791 pCurrImage = pCurrImage->pPrev)
1792 {
1793 rc = pCurrImage->Backend->pfnRead(pCurrImage->pvBackendData,
1794 uOffset, pvBuf,
1795 cbThisRead, &cbThisRead);
1796 }
1797
1798 if (rc != VERR_VD_BLOCK_FREE)
1799 {
1800 if (RT_FAILURE(rc))
1801 break;
1802 rc = vdWriteHelper(pDisk, pImageTo, uOffset, pvBuf,
1803 cbThisRead);
1804 if (RT_FAILURE(rc))
1805 break;
1806 }
1807 else
1808 rc = VINF_SUCCESS;
1809
1810 uOffset += cbThisRead;
1811 cbRemaining -= cbThisRead;
1812
1813 if (pCbProgress && pCbProgress->pfnProgress)
1814 {
1815 rc = pCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */,
1816 uOffset * 99 / cbSize,
1817 pIfProgress->pvUser);
1818 if (RT_FAILURE(rc))
1819 break;
1820 }
1821 } while (uOffset < cbSize);
1822 }
1823
1824 /* Update parent UUID so that image chain is consistent. */
1825 RTUUID Uuid;
1826 if (nImageFrom < nImageTo)
1827 {
1828 if (pImageTo->pPrev)
1829 {
1830 rc = pImageTo->Backend->pfnGetUuid(pImageTo->pPrev->pvBackendData,
1831 &Uuid);
1832 AssertRC(rc);
1833 }
1834 else
1835 RTUuidClear(&Uuid);
1836 rc = pImageTo->Backend->pfnSetParentUuid(pImageTo->pvBackendData,
1837 &Uuid);
1838 AssertRC(rc);
1839 }
1840 else
1841 {
1842 if (pImageFrom->pNext)
1843 {
1844 rc = pImageTo->Backend->pfnGetUuid(pImageTo->pvBackendData,
1845 &Uuid);
1846 AssertRC(rc);
1847 rc = pImageFrom->Backend->pfnSetParentUuid(pImageFrom->pNext,
1848 &Uuid);
1849 AssertRC(rc);
1850 }
1851 }
1852
1853 /* Make sure destination image is back to read only if necessary. */
1854 if (pImageTo != pDisk->pLast && pImageFrom != pDisk->pLast)
1855 {
1856 uOpenFlags = pImageTo->Backend->pfnGetOpenFlags(pImageTo->pvBackendData);
1857 uOpenFlags |= VD_OPEN_FLAGS_READONLY;
1858 rc = pImageTo->Backend->pfnSetOpenFlags(pImageTo->pvBackendData,
1859 uOpenFlags);
1860 if (RT_FAILURE(rc))
1861 break;
1862 }
1863
1864 /* Delete the no longer needed images. */
1865 PVDIMAGE pImg = pImageFrom, pTmp;
1866 while (pImg != pImageTo)
1867 {
1868 if (nImageFrom < nImageTo)
1869 pTmp = pImg->pNext;
1870 else
1871 pTmp = pImg->pPrev;
1872 vdRemoveImageFromList(pDisk, pImg);
1873 pImg->Backend->pfnClose(pImg->pvBackendData, true);
1874 RTMemFree(pImg->pszFilename);
1875 RTMemFree(pImg);
1876 pImg = pTmp;
1877 }
1878 } while (0);
1879
1880 if (pvBuf)
1881 RTMemTmpFree(pvBuf);
1882
1883 if (RT_SUCCESS(rc) && pCbProgress && pCbProgress->pfnProgress)
1884 pCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */, 100,
1885 pIfProgress->pvUser);
1886
1887 LogFlowFunc(("returns %Rrc\n", rc));
1888 return rc;
1889}
1890
1891/**
1892 * Copies an image from one HDD container to another.
1893 * The copy is opened in the target HDD container.
1894 * It is possible to convert between different image formats, because the
1895 * backend for the destination may be different from the source.
1896 * If both the source and destination reference the same HDD container,
1897 * then the image is moved (by copying/deleting or renaming) to the new location.
1898 * The source container is unchanged if the move operation fails, otherwise
1899 * the image at the new location is opened in the same way as the old one was.
1900 *
1901 * @returns VBox status code.
1902 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
1903 * @param pDiskFrom Pointer to source HDD container.
1904 * @param nImage Image number, counts from 0. 0 is always base image of container.
1905 * @param pDiskTo Pointer to destination HDD container.
1906 * @param pszBackend Name of the image file backend to use.
1907 * @param pszFilename New name of the image (may be NULL if pDiskFrom == pDiskTo).
1908 * @param fMoveByRename If true, attempt to perform a move by renaming (if successful the new size is ignored).
1909 * @param cbSize New image size (0 means leave unchanged).
1910 * @param uImageFlags Flags specifying special destination image features.
1911 * @param pDstUuid New UUID of the destination image. If NULL, a new UUID is created.
1912 * This parameter is used if and only if a true copy is created.
1913 * In all rename/move cases the UUIDs are copied over.
1914 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
1915 * @param pDstVDIfsImage Pointer to the per-image VD interface list, for the
1916 * destination image.
1917 * @param pDstVDIfsOperation Pointer to the per-image VD interface list,
1918 * for the destination image.
1919 */
1920VBOXDDU_DECL(int) VDCopy(PVBOXHDD pDiskFrom, unsigned nImage, PVBOXHDD pDiskTo,
1921 const char *pszBackend, const char *pszFilename,
1922 bool fMoveByRename, uint64_t cbSize,
1923 unsigned uImageFlags, PCRTUUID pDstUuid,
1924 PVDINTERFACE pVDIfsOperation,
1925 PVDINTERFACE pDstVDIfsImage,
1926 PVDINTERFACE pDstVDIfsOperation)
1927{
1928 int rc, rc2 = VINF_SUCCESS;
1929 void *pvBuf = NULL;
1930 PVDIMAGE pImageTo = NULL;
1931
1932 LogFlowFunc(("pDiskFrom=%#p nImage=%u pDiskTo=%#p pszBackend=\"%s\" pszFilename=\"%s\" fMoveByRename=%d cbSize=%llu pVDIfsOperation=%#p pDstVDIfsImage=%#p pDstVDIfsOperation=%#p\n",
1933 pDiskFrom, nImage, pDiskTo, pszBackend, pszFilename, fMoveByRename, cbSize, pVDIfsOperation, pDstVDIfsImage, pDstVDIfsOperation));
1934
1935 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
1936 VDINTERFACETYPE_PROGRESS);
1937 PVDINTERFACEPROGRESS pCbProgress = NULL;
1938 if (pIfProgress)
1939 pCbProgress = VDGetInterfaceProgress(pIfProgress);
1940
1941 PVDINTERFACE pDstIfProgress = VDInterfaceGet(pDstVDIfsOperation,
1942 VDINTERFACETYPE_PROGRESS);
1943 PVDINTERFACEPROGRESS pDstCbProgress = NULL;
1944 if (pDstIfProgress)
1945 pDstCbProgress = VDGetInterfaceProgress(pDstIfProgress);
1946
1947 do {
1948 /* Check arguments. */
1949 AssertMsgBreakStmt(VALID_PTR(pDiskFrom), ("pDiskFrom=%#p\n", pDiskFrom),
1950 rc = VERR_INVALID_PARAMETER);
1951 AssertMsg(pDiskFrom->u32Signature == VBOXHDDDISK_SIGNATURE,
1952 ("u32Signature=%08x\n", pDiskFrom->u32Signature));
1953
1954 PVDIMAGE pImageFrom = vdGetImageByNumber(pDiskFrom, nImage);
1955 AssertPtrBreakStmt(pImageFrom, rc = VERR_VD_IMAGE_NOT_FOUND);
1956 AssertMsgBreakStmt(VALID_PTR(pDiskTo), ("pDiskTo=%#p\n", pDiskTo),
1957 rc = VERR_INVALID_PARAMETER);
1958 AssertMsg(pDiskTo->u32Signature == VBOXHDDDISK_SIGNATURE,
1959 ("u32Signature=%08x\n", pDiskTo->u32Signature));
1960
1961 /* Move the image. */
1962 if (pDiskFrom == pDiskTo)
1963 {
1964 /* Rename only works when backends are the same. */
1965 if ( fMoveByRename
1966 && !RTStrICmp(pszBackend, pImageFrom->Backend->pszBackendName))
1967 {
1968 rc = pImageFrom->Backend->pfnRename(pImageFrom->pvBackendData, pszFilename ? pszFilename : pImageFrom->pszFilename);
1969 break;
1970 }
1971
1972 /** @todo Moving (including shrinking/growing) of the image is
1973 * requested, but the rename attempt failed or it wasn't possible.
1974 * Must now copy image to temp location. */
1975 AssertReleaseMsgFailed(("VDCopy: moving by copy/delete not implemented\n"));
1976 }
1977
1978 /* pszFilename is allowed to be NULL, as this indicates copy to the existing image. */
1979 AssertMsgBreakStmt(pszFilename == NULL || (VALID_PTR(pszFilename) && *pszFilename),
1980 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
1981 rc = VERR_INVALID_PARAMETER);
1982
1983 uint64_t cbSizeFrom;
1984 cbSizeFrom = pImageFrom->Backend->pfnGetSize(pImageFrom->pvBackendData);
1985 if (cbSizeFrom == 0)
1986 {
1987 rc = VERR_VD_VALUE_NOT_FOUND;
1988 break;
1989 }
1990
1991 PDMMEDIAGEOMETRY PCHSGeometryFrom = {0, 0, 0};
1992 PDMMEDIAGEOMETRY LCHSGeometryFrom = {0, 0, 0};
1993 pImageFrom->Backend->pfnGetPCHSGeometry(pImageFrom->pvBackendData, &PCHSGeometryFrom);
1994 pImageFrom->Backend->pfnGetLCHSGeometry(pImageFrom->pvBackendData, &LCHSGeometryFrom);
1995
1996 RTUUID ImageUuid, ImageModificationUuid;
1997 RTUUID ParentUuid, ParentModificationUuid;
1998 if (pDiskFrom != pDiskTo)
1999 {
2000 if (pDstUuid)
2001 ImageUuid = *pDstUuid;
2002 else
2003 RTUuidCreate(&ImageUuid);
2004 }
2005 else
2006 {
2007 rc = pImageFrom->Backend->pfnGetUuid(pImageFrom->pvBackendData, &ImageUuid);
2008 if (RT_FAILURE(rc))
2009 RTUuidCreate(&ImageUuid);
2010 }
2011 rc = pImageFrom->Backend->pfnGetModificationUuid(pImageFrom->pvBackendData, &ImageModificationUuid);
2012 if (RT_FAILURE(rc))
2013 RTUuidClear(&ImageModificationUuid);
2014 rc = pImageFrom->Backend->pfnGetParentUuid(pImageFrom->pvBackendData, &ParentUuid);
2015 if (RT_FAILURE(rc))
2016 RTUuidClear(&ParentUuid);
2017 rc = pImageFrom->Backend->pfnGetParentModificationUuid(pImageFrom->pvBackendData, &ParentModificationUuid);
2018 if (RT_FAILURE(rc))
2019 RTUuidClear(&ParentModificationUuid);
2020
2021 char szComment[1024];
2022 rc = pImageFrom->Backend->pfnGetComment(pImageFrom->pvBackendData, szComment, sizeof(szComment));
2023 if (RT_FAILURE(rc))
2024 szComment[0] = '\0';
2025 else
2026 szComment[sizeof(szComment) - 1] = '\0';
2027
2028 unsigned uOpenFlagsFrom;
2029 uOpenFlagsFrom = pImageFrom->Backend->pfnGetOpenFlags(pImageFrom->pvBackendData);
2030
2031 if (pszFilename)
2032 {
2033 if (cbSize == 0)
2034 cbSize = cbSizeFrom;
2035
2036 /* Create destination image with the properties of the source image. */
2037 /** @todo replace the VDCreateDiff/VDCreateBase calls by direct
2038 * calls to the backend. Unifies the code and reduces the API
2039 * dependencies. */
2040 if (uImageFlags & VD_IMAGE_FLAGS_DIFF)
2041 {
2042 rc = VDCreateDiff(pDiskTo, pszBackend, pszFilename, uImageFlags,
2043 szComment, &ImageUuid, &ParentUuid, uOpenFlagsFrom & ~VD_OPEN_FLAGS_READONLY, NULL, NULL);
2044 } else {
2045 /** @todo Please, review this! It's an ugly hack I think... */
2046 if (!RTStrICmp(pszBackend, "RAW"))
2047 uImageFlags |= VD_IMAGE_FLAGS_FIXED;
2048
2049 rc = VDCreateBase(pDiskTo, pszBackend, pszFilename, cbSize,
2050 uImageFlags, szComment,
2051 &PCHSGeometryFrom, &LCHSGeometryFrom,
2052 NULL, uOpenFlagsFrom & ~VD_OPEN_FLAGS_READONLY, NULL, NULL);
2053 if (RT_SUCCESS(rc) && !RTUuidIsNull(&ImageUuid))
2054 pDiskTo->pLast->Backend->pfnSetUuid(pDiskTo->pLast->pvBackendData, &ImageUuid);
2055 if (RT_SUCCESS(rc) && !RTUuidIsNull(&ParentUuid))
2056 pDiskTo->pLast->Backend->pfnSetParentUuid(pDiskTo->pLast->pvBackendData, &ParentUuid);
2057 }
2058 if (RT_FAILURE(rc))
2059 break;
2060
2061 pImageTo = pDiskTo->pLast;
2062 AssertPtrBreakStmt(pImageTo, rc = VERR_VD_IMAGE_NOT_FOUND);
2063
2064 cbSize = RT_MIN(cbSize, cbSizeFrom);
2065 }
2066 else
2067 {
2068 pImageTo = pDiskTo->pLast;
2069 AssertPtrBreakStmt(pImageTo, rc = VERR_VD_IMAGE_NOT_FOUND);
2070
2071 uint64_t cbSizeTo;
2072 cbSizeTo = pImageTo->Backend->pfnGetSize(pImageTo->pvBackendData);
2073 if (cbSizeTo == 0)
2074 {
2075 rc = VERR_VD_VALUE_NOT_FOUND;
2076 break;
2077 }
2078
2079 if (cbSize == 0)
2080 cbSize = RT_MIN(cbSizeFrom, cbSizeTo);
2081 }
2082
2083 /* Allocate tmp buffer. */
2084 pvBuf = RTMemTmpAlloc(VD_MERGE_BUFFER_SIZE);
2085 if (!pvBuf)
2086 {
2087 rc = VERR_NO_MEMORY;
2088 break;
2089 }
2090
2091 /* Copy the data. */
2092 uint64_t uOffset = 0;
2093 uint64_t cbRemaining = cbSize;
2094
2095 do
2096 {
2097 size_t cbThisRead = RT_MIN(VD_MERGE_BUFFER_SIZE, cbRemaining);
2098
2099 rc = vdReadHelper(pDiskFrom, pImageFrom, uOffset, pvBuf,
2100 cbThisRead);
2101 if (RT_FAILURE(rc))
2102 break;
2103
2104 rc = vdWriteHelper(pDiskTo, pImageTo, uOffset, pvBuf,
2105 cbThisRead);
2106 if (RT_FAILURE(rc))
2107 break;
2108
2109 uOffset += cbThisRead;
2110 cbRemaining -= cbThisRead;
2111
2112 if (pCbProgress && pCbProgress->pfnProgress)
2113 {
2114 rc = pCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */,
2115 uOffset * 99 / cbSize,
2116 pIfProgress->pvUser);
2117 if (RT_FAILURE(rc))
2118 break;
2119 }
2120 if (pDstCbProgress && pDstCbProgress->pfnProgress)
2121 {
2122 rc = pDstCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */,
2123 uOffset * 99 / cbSize,
2124 pDstIfProgress->pvUser);
2125 if (RT_FAILURE(rc))
2126 break;
2127 }
2128 } while (uOffset < cbSize);
2129
2130 if (RT_SUCCESS(rc))
2131 {
2132 pImageTo->Backend->pfnSetModificationUuid(pImageTo->pvBackendData, &ImageModificationUuid);
2133 /** @todo double-check this - it makes little sense to copy over the parent modification uuid,
2134 * as the destination image can have a totally different parent. */
2135#if 0
2136 pImageTo->Backend->pfnSetParentModificationUuid(pImageTo->pvBackendData, &ParentModificationUuid);
2137#endif
2138 }
2139 } while (0);
2140
2141 if (RT_FAILURE(rc) && pImageTo && pszFilename)
2142 {
2143 /* Error detected, but new image created. Remove image from list. */
2144 vdRemoveImageFromList(pDiskTo, pImageTo);
2145
2146 /* Close and delete image. */
2147 rc2 = pImageTo->Backend->pfnClose(pImageTo->pvBackendData, true);
2148 AssertRC(rc2);
2149 pImageTo->pvBackendData = NULL;
2150
2151 /* Free remaining resources. */
2152 if (pImageTo->pszFilename)
2153 RTStrFree(pImageTo->pszFilename);
2154
2155 RTMemFree(pImageTo);
2156 }
2157
2158 if (pvBuf)
2159 RTMemTmpFree(pvBuf);
2160
2161 if (RT_SUCCESS(rc))
2162 {
2163 if (pCbProgress && pCbProgress->pfnProgress)
2164 pCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */, 100,
2165 pIfProgress->pvUser);
2166 if (pDstCbProgress && pDstCbProgress->pfnProgress)
2167 pDstCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */, 100,
2168 pDstIfProgress->pvUser);
2169 }
2170
2171 LogFlowFunc(("returns %Rrc\n", rc));
2172 return rc;
2173}
2174
2175/**
2176 * Optimizes the storage consumption of an image. Typically the unused blocks
2177 * have to be wiped with zeroes to achieve a substantial reduced storage use.
2178 * Another optimization done is reordering the image blocks, which can provide
2179 * a significant performance boost, as reads and writes tend to use less random
2180 * file offsets.
2181 *
2182 * @return VBox status code.
2183 * @return VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
2184 * @return VERR_VD_IMAGE_READ_ONLY if image is not writable.
2185 * @return VERR_NOT_SUPPORTED if this kind of image can be compacted, but
2186 * the code for this isn't implemented yet.
2187 * @param pDisk Pointer to HDD container.
2188 * @param nImage Image number, counts from 0. 0 is always base image of container.
2189 * @param pVDIfsOperation Pointer to the per-operation VD interface list.
2190 */
2191VBOXDDU_DECL(int) VDCompact(PVBOXHDD pDisk, unsigned nImage,
2192 PVDINTERFACE pVDIfsOperation)
2193{
2194 int rc;
2195 void *pvBuf = NULL;
2196 void *pvTmp = NULL;
2197
2198 LogFlowFunc(("pDisk=%#p nImage=%u pVDIfsOperation=%#p\n",
2199 pDisk, nImage, pVDIfsOperation));
2200
2201 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
2202 VDINTERFACETYPE_PROGRESS);
2203 PVDINTERFACEPROGRESS pCbProgress = NULL;
2204 if (pIfProgress)
2205 pCbProgress = VDGetInterfaceProgress(pIfProgress);
2206
2207 do {
2208 /* Check arguments. */
2209 AssertMsgBreakStmt(VALID_PTR(pDisk), ("pDisk=%#p\n", pDisk),
2210 rc = VERR_INVALID_PARAMETER);
2211 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE,
2212 ("u32Signature=%08x\n", pDisk->u32Signature));
2213
2214 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
2215 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
2216
2217 /* If there is no compact callback for not file based backends then
2218 * the backend doesn't need compaction. No need to make much fuss about
2219 * this. For file based ones signal this as not yet supported. */
2220 if (!pImage->Backend->pfnCompact)
2221 {
2222 if (pImage->Backend->uBackendCaps & VD_CAP_FILE)
2223 rc = VERR_NOT_SUPPORTED;
2224 else
2225 rc = VINF_SUCCESS;
2226 break;
2227 }
2228
2229 /* Insert interface for reading parent state into per-operation list,
2230 * if there is a parent image. */
2231 VDINTERFACE IfOpParent;
2232 VDINTERFACEPARENTSTATE ParentCb;
2233 VDPARENTSTATEDESC ParentUser;
2234 if (pImage->pPrev)
2235 {
2236 ParentCb.cbSize = sizeof(ParentCb);
2237 ParentCb.enmInterface = VDINTERFACETYPE_PARENTSTATE;
2238 ParentCb.pfnParentRead = vdParentRead;
2239 ParentUser.pDisk = pDisk;
2240 ParentUser.pImage = pImage->pPrev;
2241 rc = VDInterfaceAdd(&IfOpParent, "VDCompact_ParentState", VDINTERFACETYPE_PARENTSTATE,
2242 &ParentCb, &ParentUser, &pVDIfsOperation);
2243 AssertRC(rc);
2244 }
2245
2246 rc = pImage->Backend->pfnCompact(pImage->pvBackendData,
2247 0, 99,
2248 pVDIfsOperation);
2249 } while (0);
2250
2251 if (pvBuf)
2252 RTMemTmpFree(pvBuf);
2253 if (pvTmp)
2254 RTMemTmpFree(pvTmp);
2255
2256 if (RT_SUCCESS(rc))
2257 {
2258 if (pCbProgress && pCbProgress->pfnProgress)
2259 pCbProgress->pfnProgress(NULL /* WARNING! pVM=NULL */, 100,
2260 pIfProgress->pvUser);
2261 }
2262
2263 LogFlowFunc(("returns %Rrc\n", rc));
2264 return rc;
2265}
2266
2267/**
2268 * Closes the last opened image file in HDD container.
2269 * If previous image file was opened in read-only mode (that is normal) and closing image
2270 * was opened in read-write mode (the whole disk was in read-write mode) - the previous image
2271 * will be reopened in read/write mode.
2272 *
2273 * @returns VBox status code.
2274 * @returns VERR_VD_NOT_OPENED if no image is opened in HDD container.
2275 * @param pDisk Pointer to HDD container.
2276 * @param fDelete If true, delete the image from the host disk.
2277 */
2278VBOXDDU_DECL(int) VDClose(PVBOXHDD pDisk, bool fDelete)
2279{
2280 int rc = VINF_SUCCESS;
2281
2282 LogFlowFunc(("pDisk=%#p fDelete=%d\n", pDisk, fDelete));
2283 do
2284 {
2285 /* sanity check */
2286 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2287 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2288
2289 PVDIMAGE pImage = pDisk->pLast;
2290 if (!pImage)
2291 {
2292 rc = VERR_VD_NOT_OPENED;
2293 break;
2294 }
2295 unsigned uOpenFlags = pImage->Backend->pfnGetOpenFlags(pImage->pvBackendData);
2296 /* Remove image from list of opened images. */
2297 vdRemoveImageFromList(pDisk, pImage);
2298 /* Close (and optionally delete) image. */
2299 rc = pImage->Backend->pfnClose(pImage->pvBackendData, fDelete);
2300 /* Free remaining resources related to the image. */
2301 RTStrFree(pImage->pszFilename);
2302 RTMemFree(pImage);
2303
2304 pImage = pDisk->pLast;
2305 if (!pImage)
2306 break;
2307
2308 /* If disk was previously in read/write mode, make sure it will stay
2309 * like this (if possible) after closing this image. Set the open flags
2310 * accordingly. */
2311 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY))
2312 {
2313 uOpenFlags = pImage->Backend->pfnGetOpenFlags(pImage->pvBackendData);
2314 uOpenFlags &= ~ VD_OPEN_FLAGS_READONLY;
2315 rc = pImage->Backend->pfnSetOpenFlags(pImage->pvBackendData, uOpenFlags);
2316 }
2317
2318 int rc2;
2319
2320 /* Cache disk information. */
2321 pDisk->cbSize = pImage->Backend->pfnGetSize(pImage->pvBackendData);
2322
2323 /* Cache PCHS geometry. */
2324 rc2 = pImage->Backend->pfnGetPCHSGeometry(pImage->pvBackendData,
2325 &pDisk->PCHSGeometry);
2326 if (RT_FAILURE(rc2))
2327 {
2328 pDisk->PCHSGeometry.cCylinders = 0;
2329 pDisk->PCHSGeometry.cHeads = 0;
2330 pDisk->PCHSGeometry.cSectors = 0;
2331 }
2332 else
2333 {
2334 /* Make sure the PCHS geometry is properly clipped. */
2335 pDisk->PCHSGeometry.cCylinders = RT_MIN(pDisk->PCHSGeometry.cCylinders, 16383);
2336 pDisk->PCHSGeometry.cHeads = RT_MIN(pDisk->PCHSGeometry.cHeads, 16);
2337 pDisk->PCHSGeometry.cSectors = RT_MIN(pDisk->PCHSGeometry.cSectors, 63);
2338 }
2339
2340 /* Cache LCHS geometry. */
2341 rc2 = pImage->Backend->pfnGetLCHSGeometry(pImage->pvBackendData,
2342 &pDisk->LCHSGeometry);
2343 if (RT_FAILURE(rc2))
2344 {
2345 pDisk->LCHSGeometry.cCylinders = 0;
2346 pDisk->LCHSGeometry.cHeads = 0;
2347 pDisk->LCHSGeometry.cSectors = 0;
2348 }
2349 else
2350 {
2351 /* Make sure the LCHS geometry is properly clipped. */
2352 pDisk->LCHSGeometry.cHeads = RT_MIN(pDisk->LCHSGeometry.cHeads, 255);
2353 pDisk->LCHSGeometry.cSectors = RT_MIN(pDisk->LCHSGeometry.cSectors, 63);
2354 }
2355 } while (0);
2356
2357 LogFlowFunc(("returns %Rrc\n", rc));
2358 return rc;
2359}
2360
2361/**
2362 * Closes all opened image files in HDD container.
2363 *
2364 * @returns VBox status code.
2365 * @param pDisk Pointer to HDD container.
2366 */
2367VBOXDDU_DECL(int) VDCloseAll(PVBOXHDD pDisk)
2368{
2369 int rc = VINF_SUCCESS;
2370
2371 LogFlowFunc(("pDisk=%#p\n", pDisk));
2372 do
2373 {
2374 /* sanity check */
2375 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2376 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2377
2378 PVDIMAGE pImage = pDisk->pLast;
2379 while (VALID_PTR(pImage))
2380 {
2381 PVDIMAGE pPrev = pImage->pPrev;
2382 /* Remove image from list of opened images. */
2383 vdRemoveImageFromList(pDisk, pImage);
2384 /* Close image. */
2385 int rc2 = pImage->Backend->pfnClose(pImage->pvBackendData, false);
2386 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
2387 rc = rc2;
2388 /* Free remaining resources related to the image. */
2389 RTStrFree(pImage->pszFilename);
2390 RTMemFree(pImage);
2391 pImage = pPrev;
2392 }
2393 Assert(!VALID_PTR(pDisk->pLast));
2394 } while (0);
2395
2396 LogFlowFunc(("returns %Rrc\n", rc));
2397 return rc;
2398}
2399
2400/**
2401 * Read data from virtual HDD.
2402 *
2403 * @returns VBox status code.
2404 * @returns VERR_VD_NOT_OPENED if no image is opened in HDD container.
2405 * @param pDisk Pointer to HDD container.
2406 * @param uOffset Offset of first reading byte from start of disk.
2407 * @param pvBuf Pointer to buffer for reading data.
2408 * @param cbRead Number of bytes to read.
2409 */
2410VBOXDDU_DECL(int) VDRead(PVBOXHDD pDisk, uint64_t uOffset, void *pvBuf,
2411 size_t cbRead)
2412{
2413 int rc;
2414
2415 LogFlowFunc(("pDisk=%#p uOffset=%llu pvBuf=%p cbRead=%zu\n",
2416 pDisk, uOffset, pvBuf, cbRead));
2417 do
2418 {
2419 /* sanity check */
2420 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2421 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2422
2423 /* Check arguments. */
2424 AssertMsgBreakStmt(VALID_PTR(pvBuf),
2425 ("pvBuf=%#p\n", pvBuf),
2426 rc = VERR_INVALID_PARAMETER);
2427 AssertMsgBreakStmt(cbRead,
2428 ("cbRead=%zu\n", cbRead),
2429 rc = VERR_INVALID_PARAMETER);
2430 AssertMsgBreakStmt(uOffset + cbRead <= pDisk->cbSize,
2431 ("uOffset=%llu cbRead=%zu pDisk->cbSize=%llu\n",
2432 uOffset, cbRead, pDisk->cbSize),
2433 rc = VERR_INVALID_PARAMETER);
2434
2435 PVDIMAGE pImage = pDisk->pLast;
2436 AssertPtrBreakStmt(pImage, rc = VERR_VD_NOT_OPENED);
2437
2438 rc = vdReadHelper(pDisk, pImage, uOffset, pvBuf, cbRead);
2439 } while (0);
2440
2441 LogFlowFunc(("returns %Rrc\n", rc));
2442 return rc;
2443}
2444
2445/**
2446 * Write data to virtual HDD.
2447 *
2448 * @returns VBox status code.
2449 * @returns VERR_VD_NOT_OPENED if no image is opened in HDD container.
2450 * @param pDisk Pointer to HDD container.
2451 * @param uOffset Offset of the first byte being
2452 * written from start of disk.
2453 * @param pvBuf Pointer to buffer for writing data.
2454 * @param cbWrite Number of bytes to write.
2455 */
2456VBOXDDU_DECL(int) VDWrite(PVBOXHDD pDisk, uint64_t uOffset, const void *pvBuf,
2457 size_t cbWrite)
2458{
2459 int rc = VINF_SUCCESS;
2460
2461 LogFlowFunc(("pDisk=%#p uOffset=%llu pvBuf=%p cbWrite=%zu\n",
2462 pDisk, uOffset, pvBuf, cbWrite));
2463 do
2464 {
2465 /* sanity check */
2466 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2467 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2468
2469 /* Check arguments. */
2470 AssertMsgBreakStmt(VALID_PTR(pvBuf),
2471 ("pvBuf=%#p\n", pvBuf),
2472 rc = VERR_INVALID_PARAMETER);
2473 AssertMsgBreakStmt(cbWrite,
2474 ("cbWrite=%zu\n", cbWrite),
2475 rc = VERR_INVALID_PARAMETER);
2476 AssertMsgBreakStmt(uOffset + cbWrite <= pDisk->cbSize,
2477 ("uOffset=%llu cbWrite=%zu pDisk->cbSize=%llu\n",
2478 uOffset, cbWrite, pDisk->cbSize),
2479 rc = VERR_INVALID_PARAMETER);
2480
2481 PVDIMAGE pImage = pDisk->pLast;
2482 AssertPtrBreakStmt(pImage, rc = VERR_VD_NOT_OPENED);
2483
2484 vdSetModifiedFlag(pDisk);
2485 rc = vdWriteHelper(pDisk, pImage, uOffset, pvBuf, cbWrite);
2486 } while (0);
2487
2488 LogFlowFunc(("returns %Rrc\n", rc));
2489 return rc;
2490}
2491
2492/**
2493 * Make sure the on disk representation of a virtual HDD is up to date.
2494 *
2495 * @returns VBox status code.
2496 * @returns VERR_VD_NOT_OPENED if no image is opened in HDD container.
2497 * @param pDisk Pointer to HDD container.
2498 */
2499VBOXDDU_DECL(int) VDFlush(PVBOXHDD pDisk)
2500{
2501 int rc = VINF_SUCCESS;
2502
2503 LogFlowFunc(("pDisk=%#p\n", pDisk));
2504 do
2505 {
2506 /* sanity check */
2507 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2508 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2509
2510 PVDIMAGE pImage = pDisk->pLast;
2511 AssertPtrBreakStmt(pImage, rc = VERR_VD_NOT_OPENED);
2512
2513 vdResetModifiedFlag(pDisk);
2514 rc = pImage->Backend->pfnFlush(pImage->pvBackendData);
2515 } while (0);
2516
2517 LogFlowFunc(("returns %Rrc\n", rc));
2518 return rc;
2519}
2520
2521/**
2522 * Get number of opened images in HDD container.
2523 *
2524 * @returns Number of opened images for HDD container. 0 if no images have been opened.
2525 * @param pDisk Pointer to HDD container.
2526 */
2527VBOXDDU_DECL(unsigned) VDGetCount(PVBOXHDD pDisk)
2528{
2529 unsigned cImages;
2530
2531 LogFlowFunc(("pDisk=%#p\n", pDisk));
2532 do
2533 {
2534 /* sanity check */
2535 AssertPtrBreakStmt(pDisk, cImages = 0);
2536 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2537
2538 cImages = pDisk->cImages;
2539 } while (0);
2540
2541 LogFlowFunc(("returns %u\n", cImages));
2542 return cImages;
2543}
2544
2545/**
2546 * Get read/write mode of HDD container.
2547 *
2548 * @returns Virtual disk ReadOnly status.
2549 * @returns true if no image is opened in HDD container.
2550 * @param pDisk Pointer to HDD container.
2551 */
2552VBOXDDU_DECL(bool) VDIsReadOnly(PVBOXHDD pDisk)
2553{
2554 bool fReadOnly;
2555
2556 LogFlowFunc(("pDisk=%#p\n", pDisk));
2557 do
2558 {
2559 /* sanity check */
2560 AssertPtrBreakStmt(pDisk, fReadOnly = false);
2561 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2562
2563 PVDIMAGE pImage = pDisk->pLast;
2564 AssertPtrBreakStmt(pImage, fReadOnly = true);
2565
2566 unsigned uOpenFlags;
2567 uOpenFlags = pDisk->pLast->Backend->pfnGetOpenFlags(pDisk->pLast->pvBackendData);
2568 fReadOnly = !!(uOpenFlags & VD_OPEN_FLAGS_READONLY);
2569 } while (0);
2570
2571 LogFlowFunc(("returns %d\n", fReadOnly));
2572 return fReadOnly;
2573}
2574
2575/**
2576 * Get total capacity of an image in HDD container.
2577 *
2578 * @returns Virtual disk size in bytes.
2579 * @returns 0 if no image with specified number was not opened.
2580 * @param pDisk Pointer to HDD container.
2581 * @param nImage Image number, counds from 0. 0 is always base image of container.
2582 */
2583VBOXDDU_DECL(uint64_t) VDGetSize(PVBOXHDD pDisk, unsigned nImage)
2584{
2585 uint64_t cbSize;
2586
2587 LogFlowFunc(("pDisk=%#p nImage=%u\n", pDisk, nImage));
2588 do
2589 {
2590 /* sanity check */
2591 AssertPtrBreakStmt(pDisk, cbSize = 0);
2592 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2593
2594 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
2595 AssertPtrBreakStmt(pImage, cbSize = 0);
2596 cbSize = pImage->Backend->pfnGetSize(pImage->pvBackendData);
2597 } while (0);
2598
2599 LogFlowFunc(("returns %llu\n", cbSize));
2600 return cbSize;
2601}
2602
2603/**
2604 * Get total file size of an image in HDD container.
2605 *
2606 * @returns Virtual disk size in bytes.
2607 * @returns 0 if no image is opened in HDD container.
2608 * @param pDisk Pointer to HDD container.
2609 * @param nImage Image number, counts from 0. 0 is always base image of container.
2610 */
2611VBOXDDU_DECL(uint64_t) VDGetFileSize(PVBOXHDD pDisk, unsigned nImage)
2612{
2613 uint64_t cbSize;
2614
2615 LogFlowFunc(("pDisk=%#p nImage=%u\n", pDisk, nImage));
2616 do
2617 {
2618 /* sanity check */
2619 AssertPtrBreakStmt(pDisk, cbSize = 0);
2620 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2621
2622 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
2623 AssertPtrBreakStmt(pImage, cbSize = 0);
2624 cbSize = pImage->Backend->pfnGetFileSize(pImage->pvBackendData);
2625 } while (0);
2626
2627 LogFlowFunc(("returns %llu\n", cbSize));
2628 return cbSize;
2629}
2630
2631/**
2632 * Get virtual disk PCHS geometry stored in HDD container.
2633 *
2634 * @returns VBox status code.
2635 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
2636 * @returns VERR_VD_GEOMETRY_NOT_SET if no geometry present in the HDD container.
2637 * @param pDisk Pointer to HDD container.
2638 * @param nImage Image number, counts from 0. 0 is always base image of container.
2639 * @param pPCHSGeometry Where to store PCHS geometry. Not NULL.
2640 */
2641VBOXDDU_DECL(int) VDGetPCHSGeometry(PVBOXHDD pDisk, unsigned nImage,
2642 PPDMMEDIAGEOMETRY pPCHSGeometry)
2643{
2644 int rc = VINF_SUCCESS;
2645
2646 LogFlowFunc(("pDisk=%#p nImage=%u pPCHSGeometry=%#p\n",
2647 pDisk, nImage, pPCHSGeometry));
2648 do
2649 {
2650 /* sanity check */
2651 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2652 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2653
2654 /* Check arguments. */
2655 AssertMsgBreakStmt(VALID_PTR(pPCHSGeometry),
2656 ("pPCHSGeometry=%#p\n", pPCHSGeometry),
2657 rc = VERR_INVALID_PARAMETER);
2658
2659 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
2660 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
2661
2662 if (pImage == pDisk->pLast)
2663 {
2664 /* Use cached information if possible. */
2665 if (pDisk->PCHSGeometry.cCylinders != 0)
2666 *pPCHSGeometry = pDisk->PCHSGeometry;
2667 else
2668 rc = VERR_VD_GEOMETRY_NOT_SET;
2669 }
2670 else
2671 rc = pImage->Backend->pfnGetPCHSGeometry(pImage->pvBackendData,
2672 pPCHSGeometry);
2673 } while (0);
2674
2675 LogFlowFunc(("%s: %Rrc (PCHS=%u/%u/%u)\n", __FUNCTION__, rc,
2676 pDisk->PCHSGeometry.cCylinders, pDisk->PCHSGeometry.cHeads,
2677 pDisk->PCHSGeometry.cSectors));
2678 return rc;
2679}
2680
2681/**
2682 * Store virtual disk PCHS geometry in HDD container.
2683 *
2684 * Note that in case of unrecoverable error all images in HDD container will be closed.
2685 *
2686 * @returns VBox status code.
2687 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
2688 * @returns VERR_VD_GEOMETRY_NOT_SET if no geometry present in the HDD container.
2689 * @param pDisk Pointer to HDD container.
2690 * @param nImage Image number, counts from 0. 0 is always base image of container.
2691 * @param pPCHSGeometry Where to load PCHS geometry from. Not NULL.
2692 */
2693VBOXDDU_DECL(int) VDSetPCHSGeometry(PVBOXHDD pDisk, unsigned nImage,
2694 PCPDMMEDIAGEOMETRY pPCHSGeometry)
2695{
2696 int rc = VINF_SUCCESS;
2697
2698 LogFlowFunc(("pDisk=%#p nImage=%u pPCHSGeometry=%#p PCHS=%u/%u/%u\n",
2699 pDisk, nImage, pPCHSGeometry, pPCHSGeometry->cCylinders,
2700 pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
2701 do
2702 {
2703 /* sanity check */
2704 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2705 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2706
2707 /* Check arguments. */
2708 AssertMsgBreakStmt( VALID_PTR(pPCHSGeometry)
2709 && pPCHSGeometry->cHeads <= 16
2710 && pPCHSGeometry->cSectors <= 63,
2711 ("pPCHSGeometry=%#p PCHS=%u/%u/%u\n", pPCHSGeometry,
2712 pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads,
2713 pPCHSGeometry->cSectors),
2714 rc = VERR_INVALID_PARAMETER);
2715
2716 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
2717 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
2718
2719 if (pImage == pDisk->pLast)
2720 {
2721 if ( pPCHSGeometry->cCylinders != pDisk->PCHSGeometry.cCylinders
2722 || pPCHSGeometry->cHeads != pDisk->PCHSGeometry.cHeads
2723 || pPCHSGeometry->cSectors != pDisk->PCHSGeometry.cSectors)
2724 {
2725 /* Only update geometry if it is changed. Avoids similar checks
2726 * in every backend. Most of the time the new geometry is set
2727 * to the previous values, so no need to go through the hassle
2728 * of updating an image which could be opened in read-only mode
2729 * right now. */
2730 rc = pImage->Backend->pfnSetPCHSGeometry(pImage->pvBackendData,
2731 pPCHSGeometry);
2732
2733 /* Cache new geometry values in any case. */
2734 int rc2 = pImage->Backend->pfnGetPCHSGeometry(pImage->pvBackendData,
2735 &pDisk->PCHSGeometry);
2736 if (RT_FAILURE(rc2))
2737 {
2738 pDisk->PCHSGeometry.cCylinders = 0;
2739 pDisk->PCHSGeometry.cHeads = 0;
2740 pDisk->PCHSGeometry.cSectors = 0;
2741 }
2742 else
2743 {
2744 /* Make sure the CHS geometry is properly clipped. */
2745 pDisk->PCHSGeometry.cHeads = RT_MIN(pDisk->PCHSGeometry.cHeads, 255);
2746 pDisk->PCHSGeometry.cSectors = RT_MIN(pDisk->PCHSGeometry.cSectors, 63);
2747 }
2748 }
2749 }
2750 else
2751 {
2752 PDMMEDIAGEOMETRY PCHS;
2753 rc = pImage->Backend->pfnGetPCHSGeometry(pImage->pvBackendData,
2754 &PCHS);
2755 if ( RT_FAILURE(rc)
2756 || pPCHSGeometry->cCylinders != PCHS.cCylinders
2757 || pPCHSGeometry->cHeads != PCHS.cHeads
2758 || pPCHSGeometry->cSectors != PCHS.cSectors)
2759 {
2760 /* Only update geometry if it is changed. Avoids similar checks
2761 * in every backend. Most of the time the new geometry is set
2762 * to the previous values, so no need to go through the hassle
2763 * of updating an image which could be opened in read-only mode
2764 * right now. */
2765 rc = pImage->Backend->pfnSetPCHSGeometry(pImage->pvBackendData,
2766 pPCHSGeometry);
2767 }
2768 }
2769 } while (0);
2770
2771 LogFlowFunc(("returns %Rrc\n", rc));
2772 return rc;
2773}
2774
2775/**
2776 * Get virtual disk LCHS geometry stored in HDD container.
2777 *
2778 * @returns VBox status code.
2779 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
2780 * @returns VERR_VD_GEOMETRY_NOT_SET if no geometry present in the HDD container.
2781 * @param pDisk Pointer to HDD container.
2782 * @param nImage Image number, counts from 0. 0 is always base image of container.
2783 * @param pLCHSGeometry Where to store LCHS geometry. Not NULL.
2784 */
2785VBOXDDU_DECL(int) VDGetLCHSGeometry(PVBOXHDD pDisk, unsigned nImage,
2786 PPDMMEDIAGEOMETRY pLCHSGeometry)
2787{
2788 int rc = VINF_SUCCESS;
2789
2790 LogFlowFunc(("pDisk=%#p nImage=%u pLCHSGeometry=%#p\n",
2791 pDisk, nImage, pLCHSGeometry));
2792 do
2793 {
2794 /* sanity check */
2795 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2796 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2797
2798 /* Check arguments. */
2799 AssertMsgBreakStmt(VALID_PTR(pLCHSGeometry),
2800 ("pLCHSGeometry=%#p\n", pLCHSGeometry),
2801 rc = VERR_INVALID_PARAMETER);
2802
2803 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
2804 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
2805
2806 if (pImage == pDisk->pLast)
2807 {
2808 /* Use cached information if possible. */
2809 if (pDisk->LCHSGeometry.cCylinders != 0)
2810 *pLCHSGeometry = pDisk->LCHSGeometry;
2811 else
2812 rc = VERR_VD_GEOMETRY_NOT_SET;
2813 }
2814 else
2815 rc = pImage->Backend->pfnGetLCHSGeometry(pImage->pvBackendData,
2816 pLCHSGeometry);
2817 } while (0);
2818
2819 LogFlowFunc((": %Rrc (LCHS=%u/%u/%u)\n", rc,
2820 pDisk->LCHSGeometry.cCylinders, pDisk->LCHSGeometry.cHeads,
2821 pDisk->LCHSGeometry.cSectors));
2822 return rc;
2823}
2824
2825/**
2826 * Store virtual disk LCHS geometry in HDD container.
2827 *
2828 * Note that in case of unrecoverable error all images in HDD container will be closed.
2829 *
2830 * @returns VBox status code.
2831 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
2832 * @returns VERR_VD_GEOMETRY_NOT_SET if no geometry present in the HDD container.
2833 * @param pDisk Pointer to HDD container.
2834 * @param nImage Image number, counts from 0. 0 is always base image of container.
2835 * @param pLCHSGeometry Where to load LCHS geometry from. Not NULL.
2836 */
2837VBOXDDU_DECL(int) VDSetLCHSGeometry(PVBOXHDD pDisk, unsigned nImage,
2838 PCPDMMEDIAGEOMETRY pLCHSGeometry)
2839{
2840 int rc = VINF_SUCCESS;
2841
2842 LogFlowFunc(("pDisk=%#p nImage=%u pLCHSGeometry=%#p LCHS=%u/%u/%u\n",
2843 pDisk, nImage, pLCHSGeometry, pLCHSGeometry->cCylinders,
2844 pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
2845 do
2846 {
2847 /* sanity check */
2848 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2849 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2850
2851 /* Check arguments. */
2852 AssertMsgBreakStmt( VALID_PTR(pLCHSGeometry)
2853 && pLCHSGeometry->cHeads <= 255
2854 && pLCHSGeometry->cSectors <= 63,
2855 ("pLCHSGeometry=%#p LCHS=%u/%u/%u\n", pLCHSGeometry,
2856 pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads,
2857 pLCHSGeometry->cSectors),
2858 rc = VERR_INVALID_PARAMETER);
2859
2860 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
2861 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
2862
2863 if (pImage == pDisk->pLast)
2864 {
2865 if ( pLCHSGeometry->cCylinders != pDisk->LCHSGeometry.cCylinders
2866 || pLCHSGeometry->cHeads != pDisk->LCHSGeometry.cHeads
2867 || pLCHSGeometry->cSectors != pDisk->LCHSGeometry.cSectors)
2868 {
2869 /* Only update geometry if it is changed. Avoids similar checks
2870 * in every backend. Most of the time the new geometry is set
2871 * to the previous values, so no need to go through the hassle
2872 * of updating an image which could be opened in read-only mode
2873 * right now. */
2874 rc = pImage->Backend->pfnSetLCHSGeometry(pImage->pvBackendData,
2875 pLCHSGeometry);
2876
2877 /* Cache new geometry values in any case. */
2878 int rc2 = pImage->Backend->pfnGetLCHSGeometry(pImage->pvBackendData,
2879 &pDisk->LCHSGeometry);
2880 if (RT_FAILURE(rc2))
2881 {
2882 pDisk->LCHSGeometry.cCylinders = 0;
2883 pDisk->LCHSGeometry.cHeads = 0;
2884 pDisk->LCHSGeometry.cSectors = 0;
2885 }
2886 else
2887 {
2888 /* Make sure the CHS geometry is properly clipped. */
2889 pDisk->LCHSGeometry.cHeads = RT_MIN(pDisk->LCHSGeometry.cHeads, 255);
2890 pDisk->LCHSGeometry.cSectors = RT_MIN(pDisk->LCHSGeometry.cSectors, 63);
2891 }
2892 }
2893 }
2894 else
2895 {
2896 PDMMEDIAGEOMETRY LCHS;
2897 rc = pImage->Backend->pfnGetLCHSGeometry(pImage->pvBackendData,
2898 &LCHS);
2899 if ( RT_FAILURE(rc)
2900 || pLCHSGeometry->cCylinders != LCHS.cCylinders
2901 || pLCHSGeometry->cHeads != LCHS.cHeads
2902 || pLCHSGeometry->cSectors != LCHS.cSectors)
2903 {
2904 /* Only update geometry if it is changed. Avoids similar checks
2905 * in every backend. Most of the time the new geometry is set
2906 * to the previous values, so no need to go through the hassle
2907 * of updating an image which could be opened in read-only mode
2908 * right now. */
2909 rc = pImage->Backend->pfnSetLCHSGeometry(pImage->pvBackendData,
2910 pLCHSGeometry);
2911 }
2912 }
2913 } while (0);
2914
2915 LogFlowFunc(("returns %Rrc\n", rc));
2916 return rc;
2917}
2918
2919/**
2920 * Get version of image in HDD container.
2921 *
2922 * @returns VBox status code.
2923 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
2924 * @param pDisk Pointer to HDD container.
2925 * @param nImage Image number, counts from 0. 0 is always base image of container.
2926 * @param puVersion Where to store the image version.
2927 */
2928VBOXDDU_DECL(int) VDGetVersion(PVBOXHDD pDisk, unsigned nImage,
2929 unsigned *puVersion)
2930{
2931 int rc = VINF_SUCCESS;
2932
2933 LogFlowFunc(("pDisk=%#p nImage=%u puVersion=%#p\n",
2934 pDisk, nImage, puVersion));
2935 do
2936 {
2937 /* sanity check */
2938 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2939 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2940
2941 /* Check arguments. */
2942 AssertMsgBreakStmt(VALID_PTR(puVersion),
2943 ("puVersion=%#p\n", puVersion),
2944 rc = VERR_INVALID_PARAMETER);
2945
2946 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
2947 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
2948
2949 *puVersion = pImage->Backend->pfnGetVersion(pImage->pvBackendData);
2950 } while (0);
2951
2952 LogFlowFunc(("returns %Rrc uVersion=%#x\n", rc, *puVersion));
2953 return rc;
2954}
2955
2956/**
2957 * List the capabilities of image backend in HDD container.
2958 *
2959 * @returns VBox status code.
2960 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
2961 * @param pDisk Pointer to the HDD container.
2962 * @param nImage Image number, counts from 0. 0 is always base image of container.
2963 * @param pbackendInfo Where to store the backend information.
2964 */
2965VBOXDDU_DECL(int) VDBackendInfoSingle(PVBOXHDD pDisk, unsigned nImage,
2966 PVDBACKENDINFO pBackendInfo)
2967{
2968 int rc = VINF_SUCCESS;
2969
2970 LogFlowFunc(("pDisk=%#p nImage=%u pBackendInfo=%#p\n",
2971 pDisk, nImage, pBackendInfo));
2972 do
2973 {
2974 /* sanity check */
2975 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
2976 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
2977
2978 /* Check arguments. */
2979 AssertMsgBreakStmt(VALID_PTR(pBackendInfo),
2980 ("pBackendInfo=%#p\n", pBackendInfo),
2981 rc = VERR_INVALID_PARAMETER);
2982
2983 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
2984 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
2985
2986 pBackendInfo->pszBackend = RTStrDup(pImage->Backend->pszBackendName);
2987 pBackendInfo->uBackendCaps = pImage->Backend->uBackendCaps;
2988 pBackendInfo->papszFileExtensions = pImage->Backend->papszFileExtensions;
2989 pBackendInfo->paConfigInfo = pImage->Backend->paConfigInfo;
2990 } while (0);
2991
2992 LogFlowFunc(("returns %Rrc\n", rc));
2993 return rc;
2994}
2995
2996/**
2997 * Get flags of image in HDD container.
2998 *
2999 * @returns VBox status code.
3000 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3001 * @param pDisk Pointer to HDD container.
3002 * @param nImage Image number, counts from 0. 0 is always base image of container.
3003 * @param puImageFlags Where to store the image flags.
3004 */
3005VBOXDDU_DECL(int) VDGetImageFlags(PVBOXHDD pDisk, unsigned nImage,
3006 unsigned *puImageFlags)
3007{
3008 int rc = VINF_SUCCESS;
3009
3010 LogFlowFunc(("pDisk=%#p nImage=%u puImageFlags=%#p\n",
3011 pDisk, nImage, puImageFlags));
3012 do
3013 {
3014 /* sanity check */
3015 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3016 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3017
3018 /* Check arguments. */
3019 AssertMsgBreakStmt(VALID_PTR(puImageFlags),
3020 ("puImageFlags=%#p\n", puImageFlags),
3021 rc = VERR_INVALID_PARAMETER);
3022
3023 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3024 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3025
3026 *puImageFlags = pImage->Backend->pfnGetImageFlags(pImage->pvBackendData);
3027 } while (0);
3028
3029 LogFlowFunc(("returns %Rrc uImageFlags=%#x\n", rc, *puImageFlags));
3030 return rc;
3031}
3032
3033/**
3034 * Get open flags of image in HDD container.
3035 *
3036 * @returns VBox status code.
3037 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3038 * @param pDisk Pointer to HDD container.
3039 * @param nImage Image number, counts from 0. 0 is always base image of container.
3040 * @param puOpenFlags Where to store the image open flags.
3041 */
3042VBOXDDU_DECL(int) VDGetOpenFlags(PVBOXHDD pDisk, unsigned nImage,
3043 unsigned *puOpenFlags)
3044{
3045 int rc = VINF_SUCCESS;
3046
3047 LogFlowFunc(("pDisk=%#p nImage=%u puOpenFlags=%#p\n",
3048 pDisk, nImage, puOpenFlags));
3049 do
3050 {
3051 /* sanity check */
3052 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3053 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3054
3055 /* Check arguments. */
3056 AssertMsgBreakStmt(VALID_PTR(puOpenFlags),
3057 ("puOpenFlags=%#p\n", puOpenFlags),
3058 rc = VERR_INVALID_PARAMETER);
3059
3060 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3061 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3062
3063 *puOpenFlags = pImage->Backend->pfnGetOpenFlags(pImage->pvBackendData);
3064 } while (0);
3065
3066 LogFlowFunc(("returns %Rrc uOpenFlags=%#x\n", rc, *puOpenFlags));
3067 return rc;
3068}
3069
3070/**
3071 * Set open flags of image in HDD container.
3072 * This operation may cause file locking changes and/or files being reopened.
3073 * Note that in case of unrecoverable error all images in HDD container will be closed.
3074 *
3075 * @returns VBox status code.
3076 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3077 * @param pDisk Pointer to HDD container.
3078 * @param nImage Image number, counts from 0. 0 is always base image of container.
3079 * @param uOpenFlags Image file open mode, see VD_OPEN_FLAGS_* constants.
3080 */
3081VBOXDDU_DECL(int) VDSetOpenFlags(PVBOXHDD pDisk, unsigned nImage,
3082 unsigned uOpenFlags)
3083{
3084 int rc;
3085
3086 LogFlowFunc(("pDisk=%#p uOpenFlags=%#u\n", pDisk, uOpenFlags));
3087 do
3088 {
3089 /* sanity check */
3090 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3091 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3092
3093 /* Check arguments. */
3094 AssertMsgBreakStmt((uOpenFlags & ~VD_OPEN_FLAGS_MASK) == 0,
3095 ("uOpenFlags=%#x\n", uOpenFlags),
3096 rc = VERR_INVALID_PARAMETER);
3097
3098 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3099 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3100
3101 rc = pImage->Backend->pfnSetOpenFlags(pImage->pvBackendData,
3102 uOpenFlags);
3103 } while (0);
3104
3105 LogFlowFunc(("returns %Rrc\n", rc));
3106 return rc;
3107}
3108
3109/**
3110 * Get base filename of image in HDD container. Some image formats use
3111 * other filenames as well, so don't use this for anything but informational
3112 * purposes.
3113 *
3114 * @returns VBox status code.
3115 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3116 * @returns VERR_BUFFER_OVERFLOW if pszFilename buffer too small to hold filename.
3117 * @param pDisk Pointer to HDD container.
3118 * @param nImage Image number, counts from 0. 0 is always base image of container.
3119 * @param pszFilename Where to store the image file name.
3120 * @param cbFilename Size of buffer pszFilename points to.
3121 */
3122VBOXDDU_DECL(int) VDGetFilename(PVBOXHDD pDisk, unsigned nImage,
3123 char *pszFilename, unsigned cbFilename)
3124{
3125 int rc;
3126
3127 LogFlowFunc(("pDisk=%#p nImage=%u pszFilename=%#p cbFilename=%u\n",
3128 pDisk, nImage, pszFilename, cbFilename));
3129 do
3130 {
3131 /* sanity check */
3132 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3133 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3134
3135 /* Check arguments. */
3136 AssertMsgBreakStmt(VALID_PTR(pszFilename) && *pszFilename,
3137 ("pszFilename=%#p \"%s\"\n", pszFilename, pszFilename),
3138 rc = VERR_INVALID_PARAMETER);
3139 AssertMsgBreakStmt(cbFilename,
3140 ("cbFilename=%u\n", cbFilename),
3141 rc = VERR_INVALID_PARAMETER);
3142
3143 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3144 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3145
3146 size_t cb = strlen(pImage->pszFilename);
3147 if (cb <= cbFilename)
3148 {
3149 strcpy(pszFilename, pImage->pszFilename);
3150 rc = VINF_SUCCESS;
3151 }
3152 else
3153 {
3154 strncpy(pszFilename, pImage->pszFilename, cbFilename - 1);
3155 pszFilename[cbFilename - 1] = '\0';
3156 rc = VERR_BUFFER_OVERFLOW;
3157 }
3158 } while (0);
3159
3160 LogFlowFunc(("returns %Rrc, pszFilename=\"%s\"\n", rc, pszFilename));
3161 return rc;
3162}
3163
3164/**
3165 * Get the comment line of image in HDD container.
3166 *
3167 * @returns VBox status code.
3168 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3169 * @returns VERR_BUFFER_OVERFLOW if pszComment buffer too small to hold comment text.
3170 * @param pDisk Pointer to HDD container.
3171 * @param nImage Image number, counts from 0. 0 is always base image of container.
3172 * @param pszComment Where to store the comment string of image. NULL is ok.
3173 * @param cbComment The size of pszComment buffer. 0 is ok.
3174 */
3175VBOXDDU_DECL(int) VDGetComment(PVBOXHDD pDisk, unsigned nImage,
3176 char *pszComment, unsigned cbComment)
3177{
3178 int rc;
3179
3180 LogFlowFunc(("pDisk=%#p nImage=%u pszComment=%#p cbComment=%u\n",
3181 pDisk, nImage, pszComment, cbComment));
3182 do
3183 {
3184 /* sanity check */
3185 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3186 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3187
3188 /* Check arguments. */
3189 AssertMsgBreakStmt(VALID_PTR(pszComment),
3190 ("pszComment=%#p \"%s\"\n", pszComment, pszComment),
3191 rc = VERR_INVALID_PARAMETER);
3192 AssertMsgBreakStmt(cbComment,
3193 ("cbComment=%u\n", cbComment),
3194 rc = VERR_INVALID_PARAMETER);
3195
3196 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3197 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3198
3199 rc = pImage->Backend->pfnGetComment(pImage->pvBackendData, pszComment,
3200 cbComment);
3201 } while (0);
3202
3203 LogFlowFunc(("returns %Rrc, pszComment=\"%s\"\n", rc, pszComment));
3204 return rc;
3205}
3206
3207/**
3208 * Changes the comment line of image in HDD container.
3209 *
3210 * @returns VBox status code.
3211 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3212 * @param pDisk Pointer to HDD container.
3213 * @param nImage Image number, counts from 0. 0 is always base image of container.
3214 * @param pszComment New comment string (UTF-8). NULL is allowed to reset the comment.
3215 */
3216VBOXDDU_DECL(int) VDSetComment(PVBOXHDD pDisk, unsigned nImage,
3217 const char *pszComment)
3218{
3219 int rc;
3220
3221 LogFlowFunc(("pDisk=%#p nImage=%u pszComment=%#p \"%s\"\n",
3222 pDisk, nImage, pszComment, pszComment));
3223 do
3224 {
3225 /* sanity check */
3226 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3227 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3228
3229 /* Check arguments. */
3230 AssertMsgBreakStmt(VALID_PTR(pszComment) || pszComment == NULL,
3231 ("pszComment=%#p \"%s\"\n", pszComment, pszComment),
3232 rc = VERR_INVALID_PARAMETER);
3233
3234 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3235 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3236
3237 rc = pImage->Backend->pfnSetComment(pImage->pvBackendData, pszComment);
3238 } while (0);
3239
3240 LogFlowFunc(("returns %Rrc\n", rc));
3241 return rc;
3242}
3243
3244
3245/**
3246 * Get UUID of image in HDD container.
3247 *
3248 * @returns VBox status code.
3249 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3250 * @param pDisk Pointer to HDD container.
3251 * @param nImage Image number, counts from 0. 0 is always base image of container.
3252 * @param pUuid Where to store the image creation UUID.
3253 */
3254VBOXDDU_DECL(int) VDGetUuid(PVBOXHDD pDisk, unsigned nImage, PRTUUID pUuid)
3255{
3256 int rc;
3257
3258 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p\n", pDisk, nImage, pUuid));
3259 do
3260 {
3261 /* sanity check */
3262 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3263 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3264
3265 /* Check arguments. */
3266 AssertMsgBreakStmt(VALID_PTR(pUuid),
3267 ("pUuid=%#p\n", pUuid),
3268 rc = VERR_INVALID_PARAMETER);
3269
3270 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3271 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3272
3273 rc = pImage->Backend->pfnGetUuid(pImage->pvBackendData, pUuid);
3274 } while (0);
3275
3276 LogFlowFunc(("returns %Rrc, Uuid={%RTuuid}\n", rc, pUuid));
3277 return rc;
3278}
3279
3280/**
3281 * Set the image's UUID. Should not be used by normal applications.
3282 *
3283 * @returns VBox status code.
3284 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3285 * @param pDisk Pointer to HDD container.
3286 * @param nImage Image number, counts from 0. 0 is always base image of container.
3287 * @param pUuid New UUID of the image. If NULL, a new UUID is created.
3288 */
3289VBOXDDU_DECL(int) VDSetUuid(PVBOXHDD pDisk, unsigned nImage, PCRTUUID pUuid)
3290{
3291 int rc;
3292
3293 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p {%RTuuid}\n",
3294 pDisk, nImage, pUuid, pUuid));
3295 do
3296 {
3297 /* sanity check */
3298 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3299 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3300
3301 AssertMsgBreakStmt(VALID_PTR(pUuid) || pUuid == NULL,
3302 ("pUuid=%#p\n", pUuid),
3303 rc = VERR_INVALID_PARAMETER);
3304
3305 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3306 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3307
3308 RTUUID Uuid;
3309 if (!pUuid)
3310 {
3311 RTUuidCreate(&Uuid);
3312 pUuid = &Uuid;
3313 }
3314 rc = pImage->Backend->pfnSetUuid(pImage->pvBackendData, pUuid);
3315 } while (0);
3316
3317 LogFlowFunc(("returns %Rrc\n", rc));
3318 return rc;
3319}
3320
3321/**
3322 * Get last modification UUID of image in HDD container.
3323 *
3324 * @returns VBox status code.
3325 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3326 * @param pDisk Pointer to HDD container.
3327 * @param nImage Image number, counts from 0. 0 is always base image of container.
3328 * @param pUuid Where to store the image modification UUID.
3329 */
3330VBOXDDU_DECL(int) VDGetModificationUuid(PVBOXHDD pDisk, unsigned nImage, PRTUUID pUuid)
3331{
3332 int rc = VINF_SUCCESS;
3333
3334 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p\n", pDisk, nImage, pUuid));
3335 do
3336 {
3337 /* sanity check */
3338 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3339 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3340
3341 /* Check arguments. */
3342 AssertMsgBreakStmt(VALID_PTR(pUuid),
3343 ("pUuid=%#p\n", pUuid),
3344 rc = VERR_INVALID_PARAMETER);
3345
3346 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3347 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3348
3349 rc = pImage->Backend->pfnGetModificationUuid(pImage->pvBackendData,
3350 pUuid);
3351 } while (0);
3352
3353 LogFlowFunc(("returns %Rrc, Uuid={%RTuuid}\n", rc, pUuid));
3354 return rc;
3355}
3356
3357/**
3358 * Set the image's last modification UUID. Should not be used by normal applications.
3359 *
3360 * @returns VBox status code.
3361 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3362 * @param pDisk Pointer to HDD container.
3363 * @param nImage Image number, counts from 0. 0 is always base image of container.
3364 * @param pUuid New modification UUID of the image. If NULL, a new UUID is created.
3365 */
3366VBOXDDU_DECL(int) VDSetModificationUuid(PVBOXHDD pDisk, unsigned nImage, PCRTUUID pUuid)
3367{
3368 int rc;
3369
3370 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p {%RTuuid}\n",
3371 pDisk, nImage, pUuid, pUuid));
3372 do
3373 {
3374 /* sanity check */
3375 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3376 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3377
3378 /* Check arguments. */
3379 AssertMsgBreakStmt(VALID_PTR(pUuid) || pUuid == NULL,
3380 ("pUuid=%#p\n", pUuid),
3381 rc = VERR_INVALID_PARAMETER);
3382
3383 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3384 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3385
3386 RTUUID Uuid;
3387 if (!pUuid)
3388 {
3389 RTUuidCreate(&Uuid);
3390 pUuid = &Uuid;
3391 }
3392 rc = pImage->Backend->pfnSetModificationUuid(pImage->pvBackendData,
3393 pUuid);
3394 } while (0);
3395
3396 LogFlowFunc(("returns %Rrc\n", rc));
3397 return rc;
3398}
3399
3400/**
3401 * Get parent UUID of image in HDD container.
3402 *
3403 * @returns VBox status code.
3404 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3405 * @param pDisk Pointer to HDD container.
3406 * @param nImage Image number, counts from 0. 0 is always base image of container.
3407 * @param pUuid Where to store the parent image UUID.
3408 */
3409VBOXDDU_DECL(int) VDGetParentUuid(PVBOXHDD pDisk, unsigned nImage,
3410 PRTUUID pUuid)
3411{
3412 int rc = VINF_SUCCESS;
3413
3414 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p\n", pDisk, nImage, pUuid));
3415 do
3416 {
3417 /* sanity check */
3418 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3419 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3420
3421 /* Check arguments. */
3422 AssertMsgBreakStmt(VALID_PTR(pUuid),
3423 ("pUuid=%#p\n", pUuid),
3424 rc = VERR_INVALID_PARAMETER);
3425
3426 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3427 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3428
3429 rc = pImage->Backend->pfnGetParentUuid(pImage->pvBackendData, pUuid);
3430 } while (0);
3431
3432 LogFlowFunc(("returns %Rrc, Uuid={%RTuuid}\n", rc, pUuid));
3433 return rc;
3434}
3435
3436/**
3437 * Set the image's parent UUID. Should not be used by normal applications.
3438 *
3439 * @returns VBox status code.
3440 * @param pDisk Pointer to HDD container.
3441 * @param nImage Image number, counts from 0. 0 is always base image of container.
3442 * @param pUuid New parent UUID of the image. If NULL, a new UUID is created.
3443 */
3444VBOXDDU_DECL(int) VDSetParentUuid(PVBOXHDD pDisk, unsigned nImage,
3445 PCRTUUID pUuid)
3446{
3447 int rc;
3448
3449 LogFlowFunc(("pDisk=%#p nImage=%u pUuid=%#p {%RTuuid}\n",
3450 pDisk, nImage, pUuid, pUuid));
3451 do
3452 {
3453 /* sanity check */
3454 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3455 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3456
3457 /* Check arguments. */
3458 AssertMsgBreakStmt(VALID_PTR(pUuid) || pUuid == NULL,
3459 ("pUuid=%#p\n", pUuid),
3460 rc = VERR_INVALID_PARAMETER);
3461
3462 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3463 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3464
3465 RTUUID Uuid;
3466 if (!pUuid)
3467 {
3468 RTUuidCreate(&Uuid);
3469 pUuid = &Uuid;
3470 }
3471 rc = pImage->Backend->pfnSetParentUuid(pImage->pvBackendData, pUuid);
3472 } while (0);
3473
3474 LogFlowFunc(("returns %Rrc\n", rc));
3475 return rc;
3476}
3477
3478
3479/**
3480 * Debug helper - dumps all opened images in HDD container into the log file.
3481 *
3482 * @param pDisk Pointer to HDD container.
3483 */
3484VBOXDDU_DECL(void) VDDumpImages(PVBOXHDD pDisk)
3485{
3486 do
3487 {
3488 /* sanity check */
3489 AssertPtrBreak(pDisk);
3490 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3491
3492 int (*pfnMessage)(void *, const char *, ...) = NULL;
3493 void *pvUser = pDisk->pInterfaceError->pvUser;
3494
3495 if (pDisk->pInterfaceErrorCallbacks && VALID_PTR(pDisk->pInterfaceErrorCallbacks->pfnMessage))
3496 pfnMessage = pDisk->pInterfaceErrorCallbacks->pfnMessage;
3497 else
3498 {
3499 pDisk->pInterfaceErrorCallbacks->pfnMessage = vdLogMessage;
3500 pfnMessage = vdLogMessage;
3501 }
3502
3503 pfnMessage(pvUser, "--- Dumping VD Disk, Images=%u\n", pDisk->cImages);
3504 for (PVDIMAGE pImage = pDisk->pBase; pImage; pImage = pImage->pNext)
3505 {
3506 pfnMessage(pvUser, "Dumping VD image \"%s\" (Backend=%s)\n",
3507 pImage->pszFilename, pImage->Backend->pszBackendName);
3508 pImage->Backend->pfnDump(pImage->pvBackendData);
3509 }
3510 } while (0);
3511}
3512
3513/**
3514 * Query if asynchronous operations are supported for this disk.
3515 *
3516 * @returns VBox status code.
3517 * @returns VERR_VD_IMAGE_NOT_FOUND if image with specified number was not opened.
3518 * @param pDisk Pointer to the HDD container.
3519 * @param nImage Image number, counts from 0. 0 is always base image of container.
3520 * @param pfAIOSupported Where to store if async IO is supported.
3521 */
3522VBOXDDU_DECL(int) VDImageIsAsyncIOSupported(PVBOXHDD pDisk, unsigned nImage, bool *pfAIOSupported)
3523{
3524 int rc = VINF_SUCCESS;
3525
3526 LogFlowFunc(("pDisk=%#p nImage=%u pfAIOSupported=%#p\n", pDisk, nImage, pfAIOSupported));
3527 do
3528 {
3529 /* sanity check */
3530 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3531 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3532
3533 /* Check arguments. */
3534 AssertMsgBreakStmt(VALID_PTR(pfAIOSupported),
3535 ("pfAIOSupported=%#p\n", pfAIOSupported),
3536 rc = VERR_INVALID_PARAMETER);
3537
3538 PVDIMAGE pImage = vdGetImageByNumber(pDisk, nImage);
3539 AssertPtrBreakStmt(pImage, rc = VERR_VD_IMAGE_NOT_FOUND);
3540
3541 if (pImage->Backend->uBackendCaps & VD_CAP_ASYNC)
3542 *pfAIOSupported = pImage->Backend->pfnIsAsyncIOSupported(pImage->pvBackendData);
3543 else
3544 *pfAIOSupported = false;
3545 } while (0);
3546
3547 LogFlowFunc(("returns %Rrc, fAIOSupported=%u\n", rc, *pfAIOSupported));
3548 return rc;
3549}
3550
3551/**
3552 * Start a asynchronous read request.
3553 *
3554 * @returns VBox status code.
3555 * @param pDisk Pointer to the HDD container.
3556 * @param uOffset The offset of the virtual disk to read from.
3557 * @param cbRead How many bytes to read.
3558 * @param paSeg Pointer to an array of segments.
3559 * @param cSeg Number of segments in the array.
3560 * @param pvUser User data which is passed on completion
3561 */
3562VBOXDDU_DECL(int) VDAsyncRead(PVBOXHDD pDisk, uint64_t uOffset, size_t cbRead,
3563 PPDMDATASEG paSeg, unsigned cSeg,
3564 void *pvUser)
3565{
3566 int rc = VERR_VD_BLOCK_FREE;
3567
3568 LogFlowFunc(("pDisk=%#p uOffset=%llu paSeg=%p cSeg=%u cbRead=%zu\n",
3569 pDisk, uOffset, paSeg, cSeg, cbRead));
3570 do
3571 {
3572 /* sanity check */
3573 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3574 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3575
3576 /* Check arguments. */
3577 AssertMsgBreakStmt(cbRead,
3578 ("cbRead=%zu\n", cbRead),
3579 rc = VERR_INVALID_PARAMETER);
3580 AssertMsgBreakStmt(uOffset + cbRead <= pDisk->cbSize,
3581 ("uOffset=%llu cbRead=%zu pDisk->cbSize=%llu\n",
3582 uOffset, cbRead, pDisk->cbSize),
3583 rc = VERR_INVALID_PARAMETER);
3584 AssertMsgBreakStmt(VALID_PTR(paSeg),
3585 ("paSeg=%#p\n", paSeg),
3586 rc = VERR_INVALID_PARAMETER);
3587 AssertMsgBreakStmt(cSeg,
3588 ("cSeg=%zu\n", cSeg),
3589 rc = VERR_INVALID_PARAMETER);
3590
3591
3592 PVDIMAGE pImage = pDisk->pLast;
3593 AssertPtrBreakStmt(pImage, rc = VERR_VD_NOT_OPENED);
3594
3595 /* @todo: This does not work for images which do not have all meta data in memory. */
3596 for (PVDIMAGE pCurrImage = pImage;
3597 pCurrImage != NULL && rc == VERR_VD_BLOCK_FREE;
3598 pCurrImage = pCurrImage->pPrev)
3599 {
3600 rc = pCurrImage->Backend->pfnAsyncRead(pCurrImage->pvBackendData,
3601 uOffset, cbRead, paSeg, cSeg,
3602 pvUser);
3603 }
3604
3605 /* No image in the chain contains the data for the block. */
3606 if (rc == VERR_VD_BLOCK_FREE)
3607 {
3608 for (unsigned i = 0; i < cSeg && (cbRead > 0); i++)
3609 {
3610 memset(paSeg[i].pvSeg, '\0', paSeg[i].cbSeg);
3611 cbRead -= paSeg[i].cbSeg;
3612 }
3613 /* Request finished without the need to enqueue a async I/O request. Tell caller. */
3614 rc = VINF_VD_ASYNC_IO_FINISHED;
3615 }
3616
3617 } while (0);
3618
3619 LogFlowFunc(("returns %Rrc\n", rc));
3620 return rc;
3621}
3622
3623
3624/**
3625 * Start a asynchronous write request.
3626 *
3627 * @returns VBox status code.
3628 * @param pDisk Pointer to the HDD container.
3629 * @param uOffset The offset of the virtual disk to write to.
3630 * @param cbWrtie How many bytes to write.
3631 * @param paSeg Pointer to an array of segments.
3632 * @param cSeg Number of segments in the array.
3633 * @param pvUser User data which is passed on completion.
3634 */
3635VBOXDDU_DECL(int) VDAsyncWrite(PVBOXHDD pDisk, uint64_t uOffset, size_t cbWrite,
3636 PPDMDATASEG paSeg, unsigned cSeg,
3637 void *pvUser)
3638{
3639 int rc;
3640
3641 LogFlowFunc(("pDisk=%#p uOffset=%llu paSeg=%p cSeg=%u cbWrite=%zu\n",
3642 pDisk, uOffset, paSeg, cSeg, cbWrite));
3643 do
3644 {
3645 /* sanity check */
3646 AssertPtrBreakStmt(pDisk, rc = VERR_INVALID_PARAMETER);
3647 AssertMsg(pDisk->u32Signature == VBOXHDDDISK_SIGNATURE, ("u32Signature=%08x\n", pDisk->u32Signature));
3648
3649 /* Check arguments. */
3650 AssertMsgBreakStmt(cbWrite,
3651 ("cbWrite=%zu\n", cbWrite),
3652 rc = VERR_INVALID_PARAMETER);
3653 AssertMsgBreakStmt(uOffset + cbWrite <= pDisk->cbSize,
3654 ("uOffset=%llu cbWrite=%zu pDisk->cbSize=%llu\n",
3655 uOffset, cbWrite, pDisk->cbSize),
3656 rc = VERR_INVALID_PARAMETER);
3657 AssertMsgBreakStmt(VALID_PTR(paSeg),
3658 ("paSeg=%#p\n", paSeg),
3659 rc = VERR_INVALID_PARAMETER);
3660 AssertMsgBreakStmt(cSeg,
3661 ("cSeg=%zu\n", cSeg),
3662 rc = VERR_INVALID_PARAMETER);
3663
3664
3665 PVDIMAGE pImage = pDisk->pLast;
3666 AssertPtrBreakStmt(pImage, rc = VERR_VD_NOT_OPENED);
3667
3668 vdSetModifiedFlag(pDisk);
3669 rc = pImage->Backend->pfnAsyncWrite(pImage->pvBackendData,
3670 uOffset, cbWrite,
3671 paSeg, cSeg, pvUser);
3672 } while (0);
3673
3674 LogFlowFunc(("returns %Rrc\n", rc));
3675 return rc;
3676
3677}
3678
3679#if 0
3680/** @copydoc VBOXHDDBACKEND::pfnComposeLocation */
3681int genericFileComposeLocation(PVDINTERFACE pConfig, char **pszLocation)
3682{
3683 return NULL;
3684}
3685
3686
3687/** @copydoc VBOXHDDBACKEND::pfnComposeName */
3688int genericFileComposeName(PVDINTERFACE pConfig, char **pszName)
3689{
3690 return NULL;
3691}
3692#endif
Note: See TracBrowser for help on using the repository browser.

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