VirtualBox

source: vbox/trunk/src/VBox/Storage/Parallels.cpp@ 63737

Last change on this file since 63737 was 63635, checked in by vboxsync, 8 years ago

Storage/QCOW,Parallels: Fix a few possible crashes possible when opening corrupted image files

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 41.6 KB
Line 
1/* $Id: Parallels.cpp 63635 2016-08-25 13:44:59Z vboxsync $ */
2/** @file
3 *
4 * Parallels hdd disk image, core code.
5 */
6
7/*
8 * Copyright (C) 2006-2016 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#define LOG_GROUP LOG_GROUP_VD_PARALLELS
20#include <VBox/vd-plugin.h>
21#include <VBox/err.h>
22
23#include <VBox/log.h>
24#include <iprt/assert.h>
25#include <iprt/mem.h>
26#include <iprt/uuid.h>
27#include <iprt/path.h>
28#include <iprt/string.h>
29#include <iprt/asm.h>
30
31#include "VDBackends.h"
32
33#define PARALLELS_HEADER_MAGIC "WithoutFreeSpace"
34#define PARALLELS_DISK_VERSION 2
35
36/** The header of the parallels disk. */
37#pragma pack(1)
38typedef struct ParallelsHeader
39{
40 /** The magic header to identify a parallels hdd image. */
41 char HeaderIdentifier[16];
42 /** The version of the disk image. */
43 uint32_t uVersion;
44 /** The number of heads the hdd has. */
45 uint32_t cHeads;
46 /** Number of cylinders. */
47 uint32_t cCylinders;
48 /** Number of sectors per track. */
49 uint32_t cSectorsPerTrack;
50 /** Number of entries in the allocation bitmap. */
51 uint32_t cEntriesInAllocationBitmap;
52 /** Total number of sectors. */
53 uint32_t cSectors;
54 /** Padding. */
55 char Padding[24];
56} ParallelsHeader;
57#pragma pack()
58
59/**
60 * Parallels image structure.
61 */
62typedef struct PARALLELSIMAGE
63{
64 /** Image file name. */
65 const char *pszFilename;
66 /** Opaque storage handle. */
67 PVDIOSTORAGE pStorage;
68
69 /** Pointer to the per-disk VD interface list. */
70 PVDINTERFACE pVDIfsDisk;
71 /** Pointer to the per-image VD interface list. */
72 PVDINTERFACE pVDIfsImage;
73 /** Error interface. */
74 PVDINTERFACEERROR pIfError;
75 /** I/O interface. */
76 PVDINTERFACEIOINT pIfIo;
77
78 /** Open flags passed by VBoxHDD layer. */
79 unsigned uOpenFlags;
80 /** Image flags defined during creation or determined during open. */
81 unsigned uImageFlags;
82 /** Total size of the image. */
83 uint64_t cbSize;
84
85 /** Physical geometry of this image. */
86 VDGEOMETRY PCHSGeometry;
87 /** Logical geometry of this image. */
88 VDGEOMETRY LCHSGeometry;
89
90 /** Pointer to the allocation bitmap. */
91 uint32_t *pAllocationBitmap;
92 /** Entries in the allocation bitmap. */
93 uint64_t cAllocationBitmapEntries;
94 /** Flag whether the allocation bitmap was changed. */
95 bool fAllocationBitmapChanged;
96 /** Current file size. */
97 uint64_t cbFileCurrent;
98} PARALLELSIMAGE, *PPARALLELSIMAGE;
99
100
101/*********************************************************************************************************************************
102* Static Variables *
103*********************************************************************************************************************************/
104
105/** NULL-terminated array of supported file extensions. */
106static const VDFILEEXTENSION s_aParallelsFileExtensions[] =
107{
108 {"hdd", VDTYPE_HDD},
109 {NULL, VDTYPE_INVALID}
110};
111
112/***************************************************
113 * Internal functions *
114 **************************************************/
115
116/**
117 * Internal. Flush image data to disk.
118 */
119static int parallelsFlushImage(PPARALLELSIMAGE pImage)
120{
121 int rc = VINF_SUCCESS;
122
123 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
124 return VINF_SUCCESS;
125
126 if ( !(pImage->uImageFlags & VD_IMAGE_FLAGS_FIXED)
127 && (pImage->fAllocationBitmapChanged))
128 {
129 pImage->fAllocationBitmapChanged = false;
130 /* Write the allocation bitmap to the file. */
131 rc = vdIfIoIntFileWriteSync(pImage->pIfIo, pImage->pStorage,
132 sizeof(ParallelsHeader), pImage->pAllocationBitmap,
133 pImage->cAllocationBitmapEntries * sizeof(uint32_t));
134 if (RT_FAILURE(rc))
135 return rc;
136 }
137
138 /* Flush file. */
139 rc = vdIfIoIntFileFlushSync(pImage->pIfIo, pImage->pStorage);
140
141 LogFlowFunc(("returns %Rrc\n", rc));
142 return rc;
143}
144
145/**
146 * Internal. Free all allocated space for representing an image except pImage,
147 * and optionally delete the image from disk.
148 */
149static int parallelsFreeImage(PPARALLELSIMAGE pImage, bool fDelete)
150{
151 int rc = VINF_SUCCESS;
152
153 /* Freeing a never allocated image (e.g. because the open failed) is
154 * not signalled as an error. After all nothing bad happens. */
155 if (pImage)
156 {
157 if (pImage->pStorage)
158 {
159 /* No point updating the file that is deleted anyway. */
160 if (!fDelete)
161 parallelsFlushImage(pImage);
162
163 rc = vdIfIoIntFileClose(pImage->pIfIo, pImage->pStorage);
164 pImage->pStorage = NULL;
165 }
166
167 if (pImage->pAllocationBitmap)
168 {
169 RTMemFree(pImage->pAllocationBitmap);
170 pImage->pAllocationBitmap = NULL;
171 }
172
173 if (fDelete && pImage->pszFilename)
174 vdIfIoIntFileDelete(pImage->pIfIo, pImage->pszFilename);
175 }
176
177 return rc;
178}
179
180static int parallelsOpenImage(PPARALLELSIMAGE pImage, unsigned uOpenFlags)
181{
182 int rc = VINF_SUCCESS;
183 ParallelsHeader parallelsHeader;
184
185 pImage->pIfError = VDIfErrorGet(pImage->pVDIfsDisk);
186 pImage->pIfIo = VDIfIoIntGet(pImage->pVDIfsImage);
187 pImage->uOpenFlags = uOpenFlags;
188 AssertPtrReturn(pImage->pIfIo, VERR_INVALID_PARAMETER);
189
190 rc = vdIfIoIntFileOpen(pImage->pIfIo, pImage->pszFilename,
191 VDOpenFlagsToFileOpenFlags(uOpenFlags,
192 false /* fCreate */),
193 &pImage->pStorage);
194 if (RT_FAILURE(rc))
195 goto out;
196
197 rc = vdIfIoIntFileGetSize(pImage->pIfIo, pImage->pStorage, &pImage->cbFileCurrent);
198 if (RT_FAILURE(rc))
199 goto out;
200 if (pImage->cbFileCurrent % 512 != 0)
201 {
202 rc = VERR_VD_PARALLELS_INVALID_HEADER;
203 goto out;
204 }
205
206 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pImage->pStorage, 0,
207 &parallelsHeader, sizeof(parallelsHeader));
208 if (RT_FAILURE(rc))
209 goto out;
210
211 if (memcmp(parallelsHeader.HeaderIdentifier, PARALLELS_HEADER_MAGIC, 16))
212 {
213 /* Check if the file has hdd as extension. It is a fixed size raw image then. */
214 char *pszSuffix = RTPathSuffix(pImage->pszFilename);
215 if (strcmp(pszSuffix, ".hdd"))
216 {
217 rc = VERR_VD_PARALLELS_INVALID_HEADER;
218 goto out;
219 }
220
221 /* This is a fixed size image. */
222 pImage->uImageFlags |= VD_IMAGE_FLAGS_FIXED;
223 pImage->cbSize = pImage->cbFileCurrent;
224
225 pImage->PCHSGeometry.cHeads = 16;
226 pImage->PCHSGeometry.cSectors = 63;
227 uint64_t cCylinders = pImage->cbSize / (512 * pImage->PCHSGeometry.cSectors * pImage->PCHSGeometry.cHeads);
228 pImage->PCHSGeometry.cCylinders = (uint32_t)cCylinders;
229 }
230 else
231 {
232 if (parallelsHeader.uVersion != PARALLELS_DISK_VERSION)
233 {
234 rc = VERR_NOT_SUPPORTED;
235 goto out;
236 }
237
238 if (parallelsHeader.cEntriesInAllocationBitmap > (1 << 30))
239 {
240 rc = VERR_NOT_SUPPORTED;
241 goto out;
242 }
243
244 Log(("cSectors=%u\n", parallelsHeader.cSectors));
245 pImage->cbSize = ((uint64_t)parallelsHeader.cSectors) * 512;
246 pImage->uImageFlags = VD_IMAGE_FLAGS_NONE;
247 pImage->cAllocationBitmapEntries = parallelsHeader.cEntriesInAllocationBitmap;
248 pImage->pAllocationBitmap = (uint32_t *)RTMemAllocZ((uint32_t)pImage->cAllocationBitmapEntries * sizeof(uint32_t));
249 if (!pImage->pAllocationBitmap)
250 {
251 rc = VERR_NO_MEMORY;
252 goto out;
253 }
254
255 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pImage->pStorage,
256 sizeof(ParallelsHeader), pImage->pAllocationBitmap,
257 pImage->cAllocationBitmapEntries * sizeof(uint32_t));
258 if (RT_FAILURE(rc))
259 goto out;
260
261 pImage->PCHSGeometry.cCylinders = parallelsHeader.cCylinders;
262 pImage->PCHSGeometry.cHeads = parallelsHeader.cHeads;
263 pImage->PCHSGeometry.cSectors = parallelsHeader.cSectorsPerTrack;
264 }
265
266out:
267 LogFlowFunc(("returns %Rrc\n", rc));
268 return rc;
269}
270
271/**
272 * Internal: Create a parallels image.
273 */
274static int parallelsCreateImage(PPARALLELSIMAGE pImage, uint64_t cbSize,
275 unsigned uImageFlags, const char *pszComment,
276 PCVDGEOMETRY pPCHSGeometry,
277 PCVDGEOMETRY pLCHSGeometry, unsigned uOpenFlags,
278 PFNVDPROGRESS pfnProgress, void *pvUser,
279 unsigned uPercentStart, unsigned uPercentSpan)
280{
281 RT_NOREF1(pszComment);
282 int rc = VINF_SUCCESS;
283 int32_t fOpen;
284
285 if (uImageFlags & VD_IMAGE_FLAGS_FIXED)
286 {
287 rc = vdIfError(pImage->pIfError, VERR_VD_INVALID_TYPE, RT_SRC_POS, N_("Parallels: cannot create fixed image '%s'. Create a raw image"), pImage->pszFilename);
288 goto out;
289 }
290
291 pImage->uOpenFlags = uOpenFlags & ~VD_OPEN_FLAGS_READONLY;
292 pImage->uImageFlags = uImageFlags;
293 pImage->PCHSGeometry = *pPCHSGeometry;
294 pImage->LCHSGeometry = *pLCHSGeometry;
295
296 if (!pImage->PCHSGeometry.cCylinders)
297 {
298 /* Set defaults. */
299 pImage->PCHSGeometry.cSectors = 63;
300 pImage->PCHSGeometry.cHeads = 16;
301 pImage->PCHSGeometry.cCylinders = pImage->cbSize / (512 * pImage->PCHSGeometry.cSectors * pImage->PCHSGeometry.cHeads);
302 }
303
304 pImage->pIfError = VDIfErrorGet(pImage->pVDIfsDisk);
305 pImage->pIfIo = VDIfIoIntGet(pImage->pVDIfsImage);
306 AssertPtrReturn(pImage->pIfIo, VERR_INVALID_PARAMETER);
307
308 /* Create image file. */
309 fOpen = VDOpenFlagsToFileOpenFlags(pImage->uOpenFlags, true /* fCreate */);
310 rc = vdIfIoIntFileOpen(pImage->pIfIo, pImage->pszFilename, fOpen, &pImage->pStorage);
311 if (RT_FAILURE(rc))
312 {
313 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("Parallels: cannot create image '%s'"), pImage->pszFilename);
314 goto out;
315 }
316
317 if (RT_SUCCESS(rc) && pfnProgress)
318 pfnProgress(pvUser, uPercentStart + uPercentSpan * 98 / 100);
319
320 /* Setup image state. */
321 pImage->cbSize = cbSize;
322 pImage->cAllocationBitmapEntries = cbSize / 512 / pImage->PCHSGeometry.cSectors;
323 if (pImage->cAllocationBitmapEntries * pImage->PCHSGeometry.cSectors * 512 < cbSize)
324 pImage->cAllocationBitmapEntries++;
325 pImage->fAllocationBitmapChanged = true;
326 pImage->cbFileCurrent = sizeof(ParallelsHeader) + pImage->cAllocationBitmapEntries * sizeof(uint32_t);
327 /* Round to next sector boundary. */
328 pImage->cbFileCurrent += 512 - pImage->cbFileCurrent % 512;
329 Assert(!(pImage->cbFileCurrent % 512));
330 pImage->pAllocationBitmap = (uint32_t *)RTMemAllocZ(pImage->cAllocationBitmapEntries * sizeof(uint32_t));
331 if (!pImage->pAllocationBitmap)
332 rc = VERR_NO_MEMORY;
333
334 if (RT_SUCCESS(rc))
335 {
336 ParallelsHeader Header;
337
338 memcpy(Header.HeaderIdentifier, PARALLELS_HEADER_MAGIC, sizeof(Header.HeaderIdentifier));
339 Header.uVersion = RT_H2LE_U32(PARALLELS_DISK_VERSION);
340 Header.cHeads = RT_H2LE_U32(pImage->PCHSGeometry.cHeads);
341 Header.cCylinders = RT_H2LE_U32(pImage->PCHSGeometry.cCylinders);
342 Header.cSectorsPerTrack = RT_H2LE_U32(pImage->PCHSGeometry.cSectors);
343 Header.cEntriesInAllocationBitmap = RT_H2LE_U32(pImage->cAllocationBitmapEntries);
344 Header.cSectors = RT_H2LE_U32(pImage->cbSize / 512);
345 memset(Header.Padding, 0, sizeof(Header.Padding));
346
347 /* Write header and allocation bitmap. */
348 rc = vdIfIoIntFileSetSize(pImage->pIfIo, pImage->pStorage, pImage->cbFileCurrent);
349 if (RT_SUCCESS(rc))
350 rc = vdIfIoIntFileWriteSync(pImage->pIfIo, pImage->pStorage, 0,
351 &Header, sizeof(Header));
352 if (RT_SUCCESS(rc))
353 rc = parallelsFlushImage(pImage); /* Writes the allocation bitmap. */
354 }
355
356out:
357 if (RT_SUCCESS(rc) && pfnProgress)
358 pfnProgress(pvUser, uPercentStart + uPercentSpan);
359
360 if (RT_FAILURE(rc))
361 parallelsFreeImage(pImage, rc != VERR_ALREADY_EXISTS);
362 return rc;
363}
364
365/** @copydoc VBOXHDDBACKEND::pfnCheckIfValid */
366static DECLCALLBACK(int) parallelsCheckIfValid(const char *pszFilename, PVDINTERFACE pVDIfsDisk,
367 PVDINTERFACE pVDIfsImage, VDTYPE *penmType)
368{
369 RT_NOREF1(pVDIfsDisk);
370 int rc;
371 PVDIOSTORAGE pStorage;
372 ParallelsHeader parallelsHeader;
373
374 PVDINTERFACEIOINT pIfIo = VDIfIoIntGet(pVDIfsImage);
375 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
376
377 rc = vdIfIoIntFileOpen(pIfIo, pszFilename,
378 VDOpenFlagsToFileOpenFlags(VD_OPEN_FLAGS_READONLY,
379 false /* fCreate */),
380 &pStorage);
381 if (RT_FAILURE(rc))
382 return rc;
383
384 rc = vdIfIoIntFileReadSync(pIfIo, pStorage, 0, &parallelsHeader,
385 sizeof(ParallelsHeader));
386 if (RT_SUCCESS(rc))
387 {
388 if ( !memcmp(parallelsHeader.HeaderIdentifier, PARALLELS_HEADER_MAGIC, 16)
389 && (parallelsHeader.uVersion == PARALLELS_DISK_VERSION))
390 rc = VINF_SUCCESS;
391 else
392 {
393 /*
394 * The image may be an fixed size image.
395 * Unfortunately fixed sized parallels images
396 * are just raw files hence no magic header to
397 * check for.
398 * The code succeeds if the file is a multiple
399 * of 512 and if the file extensions is *.hdd
400 */
401 uint64_t cbFile;
402 char *pszSuffix;
403
404 rc = vdIfIoIntFileGetSize(pIfIo, pStorage, &cbFile);
405 if (RT_FAILURE(rc) || ((cbFile % 512) != 0))
406 {
407 vdIfIoIntFileClose(pIfIo, pStorage);
408 return VERR_VD_PARALLELS_INVALID_HEADER;
409 }
410
411 pszSuffix = RTPathSuffix(pszFilename);
412 if (!pszSuffix || strcmp(pszSuffix, ".hdd"))
413 rc = VERR_VD_PARALLELS_INVALID_HEADER;
414 else
415 rc = VINF_SUCCESS;
416 }
417 }
418
419 if (RT_SUCCESS(rc))
420 *penmType = VDTYPE_HDD;
421
422 vdIfIoIntFileClose(pIfIo, pStorage);
423 return rc;
424}
425
426/** @copydoc VBOXHDDBACKEND::pfnOpen */
427static DECLCALLBACK(int) parallelsOpen(const char *pszFilename, unsigned uOpenFlags,
428 PVDINTERFACE pVDIfsDisk, PVDINTERFACE pVDIfsImage,
429 VDTYPE enmType, void **ppBackendData)
430{
431 LogFlowFunc(("pszFilename=\"%s\" uOpenFlags=%#x pVDIfsDisk=%#p pVDIfsImage=%#p enmType=%u ppBackendData=%#p\n", pszFilename, uOpenFlags, pVDIfsDisk, pVDIfsImage, enmType, ppBackendData));
432 int rc;
433 PPARALLELSIMAGE pImage;
434
435 NOREF(enmType); /**< @todo r=klaus make use of the type info. */
436
437 /* Check open flags. All valid flags are supported. */
438 if (uOpenFlags & ~VD_OPEN_FLAGS_MASK)
439 {
440 rc = VERR_INVALID_PARAMETER;
441 goto out;
442 }
443
444 /* Check remaining arguments. */
445 if ( !VALID_PTR(pszFilename)
446 || !*pszFilename)
447 {
448 rc = VERR_INVALID_PARAMETER;
449 goto out;
450 }
451
452 pImage = (PPARALLELSIMAGE)RTMemAllocZ(sizeof(PARALLELSIMAGE));
453 if (!pImage)
454 {
455 rc = VERR_NO_MEMORY;
456 goto out;
457 }
458
459 pImage->pszFilename = pszFilename;
460 pImage->pStorage = NULL;
461 pImage->pVDIfsDisk = pVDIfsDisk;
462 pImage->pVDIfsImage = pVDIfsImage;
463 pImage->fAllocationBitmapChanged = false;
464
465 rc = parallelsOpenImage(pImage, uOpenFlags);
466 if (RT_SUCCESS(rc))
467 *ppBackendData = pImage;
468 else
469 RTMemFree(pImage);
470
471out:
472 LogFlowFunc(("returns %Rrc (pBackendData=%#p)\n", rc, *ppBackendData));
473 return rc;
474}
475
476/** @copydoc VBOXHDDBACKEND::pfnCreate */
477static DECLCALLBACK(int) parallelsCreate(const char *pszFilename, uint64_t cbSize,
478 unsigned uImageFlags, const char *pszComment,
479 PCVDGEOMETRY pPCHSGeometry,
480 PCVDGEOMETRY pLCHSGeometry, PCRTUUID pUuid,
481 unsigned uOpenFlags, unsigned uPercentStart,
482 unsigned uPercentSpan, PVDINTERFACE pVDIfsDisk,
483 PVDINTERFACE pVDIfsImage,
484 PVDINTERFACE pVDIfsOperation, VDTYPE enmType,
485 void **ppBackendData)
486{
487 RT_NOREF1(pUuid);
488 LogFlowFunc(("pszFilename=\"%s\" cbSize=%llu uImageFlags=%#x pszComment=\"%s\" pPCHSGeometry=%#p pLCHSGeometry=%#p Uuid=%RTuuid uOpenFlags=%#x uPercentStart=%u uPercentSpan=%u pVDIfsDisk=%#p pVDIfsImage=%#p pVDIfsOperation=%#p enmType=%u ppBackendData=%#p",
489 pszFilename, cbSize, uImageFlags, pszComment, pPCHSGeometry, pLCHSGeometry, pUuid, uOpenFlags, uPercentStart, uPercentSpan, pVDIfsDisk, pVDIfsImage, pVDIfsOperation, enmType, ppBackendData));
490 int rc = VINF_SUCCESS;
491 PPARALLELSIMAGE pImage;
492
493 PFNVDPROGRESS pfnProgress = NULL;
494 void *pvUser = NULL;
495 PVDINTERFACEPROGRESS pIfProgress = VDIfProgressGet(pVDIfsOperation);
496 if (pIfProgress)
497 {
498 pfnProgress = pIfProgress->pfnProgress;
499 pvUser = pIfProgress->Core.pvUser;
500 }
501
502 /* Check the VD container type. */
503 if (enmType != VDTYPE_HDD)
504 {
505 rc = VERR_VD_INVALID_TYPE;
506 goto out;
507 }
508
509 /* Check open flags. All valid flags are supported. */
510 if (uOpenFlags & ~VD_OPEN_FLAGS_MASK)
511 {
512 rc = VERR_INVALID_PARAMETER;
513 goto out;
514 }
515
516 /* Check remaining arguments. */
517 if ( !VALID_PTR(pszFilename)
518 || !*pszFilename
519 || !VALID_PTR(pPCHSGeometry)
520 || !VALID_PTR(pLCHSGeometry))
521 {
522 rc = VERR_INVALID_PARAMETER;
523 goto out;
524 }
525
526 pImage = (PPARALLELSIMAGE)RTMemAllocZ(sizeof(PARALLELSIMAGE));
527 if (!pImage)
528 {
529 rc = VERR_NO_MEMORY;
530 goto out;
531 }
532 pImage->pszFilename = pszFilename;
533 pImage->pStorage = NULL;
534 pImage->pVDIfsDisk = pVDIfsDisk;
535 pImage->pVDIfsImage = pVDIfsImage;
536
537 rc = parallelsCreateImage(pImage, cbSize, uImageFlags, pszComment,
538 pPCHSGeometry, pLCHSGeometry, uOpenFlags,
539 pfnProgress, pvUser, uPercentStart, uPercentSpan);
540 if (RT_SUCCESS(rc))
541 {
542 /* So far the image is opened in read/write mode. Make sure the
543 * image is opened in read-only mode if the caller requested that. */
544 if (uOpenFlags & VD_OPEN_FLAGS_READONLY)
545 {
546 parallelsFreeImage(pImage, false);
547 rc = parallelsOpenImage(pImage, uOpenFlags);
548 if (RT_FAILURE(rc))
549 {
550 RTMemFree(pImage);
551 goto out;
552 }
553 }
554 *ppBackendData = pImage;
555 }
556 else
557 RTMemFree(pImage);
558
559out:
560 LogFlowFunc(("returns %Rrc\n", rc));
561 return rc;
562}
563
564/** @copydoc VBOXHDDBACKEND::pfnRename */
565static DECLCALLBACK(int) parallelsRename(void *pBackendData, const char *pszFilename)
566{
567 LogFlowFunc(("pBackendData=%#p pszFilename=%#p\n", pBackendData, pszFilename));
568 int rc = VINF_SUCCESS;
569 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
570
571 /* Check arguments. */
572 if ( !pImage
573 || !pszFilename
574 || !*pszFilename)
575 {
576 rc = VERR_INVALID_PARAMETER;
577 goto out;
578 }
579
580 /* Close the image. */
581 rc = parallelsFreeImage(pImage, false);
582 if (RT_FAILURE(rc))
583 goto out;
584
585 /* Rename the file. */
586 rc = vdIfIoIntFileMove(pImage->pIfIo, pImage->pszFilename, pszFilename, 0);
587 if (RT_FAILURE(rc))
588 {
589 /* The move failed, try to reopen the original image. */
590 int rc2 = parallelsOpenImage(pImage, pImage->uOpenFlags);
591 if (RT_FAILURE(rc2))
592 rc = rc2;
593
594 goto out;
595 }
596
597 /* Update pImage with the new information. */
598 pImage->pszFilename = pszFilename;
599
600 /* Open the old image with new name. */
601 rc = parallelsOpenImage(pImage, pImage->uOpenFlags);
602 if (RT_FAILURE(rc))
603 goto out;
604
605out:
606 LogFlowFunc(("returns %Rrc\n", rc));
607 return rc;
608}
609
610/** @copydoc VBOXHDDBACKEND::pfnClose */
611static DECLCALLBACK(int) parallelsClose(void *pBackendData, bool fDelete)
612{
613 LogFlowFunc(("pBackendData=%#p fDelete=%d\n", pBackendData, fDelete));
614 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
615 int rc;
616
617 rc = parallelsFreeImage(pImage, fDelete);
618 RTMemFree(pImage);
619
620 LogFlowFunc(("returns %Rrc\n", rc));
621 return rc;
622}
623
624/** @copydoc VBOXHDDBACKEND::pfnRead */
625static DECLCALLBACK(int) parallelsRead(void *pBackendData, uint64_t uOffset, size_t cbToRead,
626 PVDIOCTX pIoCtx, size_t *pcbActuallyRead)
627{
628 LogFlowFunc(("pBackendData=%#p uOffset=%llu pIoCtx=%#p cbToRead=%zu pcbActuallyRead=%#p\n",
629 pBackendData, uOffset, pIoCtx, cbToRead, pcbActuallyRead));
630 int rc = VINF_SUCCESS;
631 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
632 uint64_t uSector;
633 uint64_t uOffsetInFile;
634 uint32_t iIndexInAllocationTable;
635
636 AssertPtr(pImage);
637 Assert(uOffset % 512 == 0);
638 Assert(cbToRead % 512 == 0);
639
640 if (pImage->uImageFlags & VD_IMAGE_FLAGS_FIXED)
641 rc = vdIfIoIntFileReadUser(pImage->pIfIo, pImage->pStorage, uOffset,
642 pIoCtx, cbToRead);
643 else
644 {
645 /* Calculate offset in the real file. */
646 uSector = uOffset / 512;
647 /* One chunk in the file is always one track big. */
648 iIndexInAllocationTable = (uint32_t)(uSector / pImage->PCHSGeometry.cSectors);
649 uSector = uSector % pImage->PCHSGeometry.cSectors;
650
651 cbToRead = RT_MIN(cbToRead, (pImage->PCHSGeometry.cSectors - uSector)*512);
652
653 if (pImage->pAllocationBitmap[iIndexInAllocationTable] == 0)
654 rc = VERR_VD_BLOCK_FREE;
655 else
656 {
657 uOffsetInFile = (pImage->pAllocationBitmap[iIndexInAllocationTable] + uSector) * 512;
658 rc = vdIfIoIntFileReadUser(pImage->pIfIo, pImage->pStorage, uOffsetInFile,
659 pIoCtx, cbToRead);
660 }
661 }
662
663 *pcbActuallyRead = cbToRead;
664
665 LogFlowFunc(("returns %Rrc\n", rc));
666 return rc;
667}
668
669/** @copydoc VBOXHDDBACKEND::pfnWrite */
670static DECLCALLBACK(int) parallelsWrite(void *pBackendData, uint64_t uOffset, size_t cbToWrite,
671 PVDIOCTX pIoCtx, size_t *pcbWriteProcess, size_t *pcbPreRead,
672 size_t *pcbPostRead, unsigned fWrite)
673{
674 LogFlowFunc(("pBackendData=%#p uOffset=%llu pIoCtx=%#p cbToWrite=%zu pcbWriteProcess=%#p\n",
675 pBackendData, uOffset, pIoCtx, cbToWrite, pcbWriteProcess));
676 int rc = VINF_SUCCESS;
677 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
678 uint64_t uSector;
679 uint64_t uOffsetInFile;
680 uint32_t iIndexInAllocationTable;
681
682 AssertPtr(pImage);
683 Assert(uOffset % 512 == 0);
684 Assert(cbToWrite % 512 == 0);
685
686 if (pImage->uImageFlags & VD_IMAGE_FLAGS_FIXED)
687 rc = vdIfIoIntFileWriteUser(pImage->pIfIo, pImage->pStorage, uOffset,
688 pIoCtx, cbToWrite, NULL, NULL);
689 else
690 {
691 /* Calculate offset in the real file. */
692 uSector = uOffset / 512;
693 /* One chunk in the file is always one track big. */
694 iIndexInAllocationTable = (uint32_t)(uSector / pImage->PCHSGeometry.cSectors);
695 uSector = uSector % pImage->PCHSGeometry.cSectors;
696
697 cbToWrite = RT_MIN(cbToWrite, (pImage->PCHSGeometry.cSectors - uSector)*512);
698
699 if (pImage->pAllocationBitmap[iIndexInAllocationTable] == 0)
700 {
701 if (fWrite & VD_WRITE_NO_ALLOC)
702 {
703 *pcbPreRead = uSector * 512;
704 *pcbPostRead = pImage->PCHSGeometry.cSectors * 512 - cbToWrite - *pcbPreRead;
705
706 if (pcbWriteProcess)
707 *pcbWriteProcess = cbToWrite;
708 return VERR_VD_BLOCK_FREE;
709 }
710
711 /* Allocate new chunk in the file. */
712 Assert(uSector == 0);
713 AssertMsg(pImage->cbFileCurrent % 512 == 0, ("File size is not a multiple of 512\n"));
714 pImage->pAllocationBitmap[iIndexInAllocationTable] = (uint32_t)(pImage->cbFileCurrent / 512);
715 pImage->cbFileCurrent += pImage->PCHSGeometry.cSectors * 512;
716 pImage->fAllocationBitmapChanged = true;
717 uOffsetInFile = (uint64_t)pImage->pAllocationBitmap[iIndexInAllocationTable] * 512;
718
719 /*
720 * Write the new block at the current end of the file.
721 */
722 rc = vdIfIoIntFileWriteUser(pImage->pIfIo, pImage->pStorage,
723 uOffsetInFile, pIoCtx, cbToWrite, NULL, NULL);
724 if (RT_SUCCESS(rc) || (rc == VERR_VD_ASYNC_IO_IN_PROGRESS))
725 {
726 /* Write the changed allocation bitmap entry. */
727 /** @todo Error handling. */
728 rc = vdIfIoIntFileWriteMeta(pImage->pIfIo, pImage->pStorage,
729 sizeof(ParallelsHeader) + iIndexInAllocationTable * sizeof(uint32_t),
730 &pImage->pAllocationBitmap[iIndexInAllocationTable],
731 sizeof(uint32_t), pIoCtx,
732 NULL, NULL);
733 }
734
735 *pcbPreRead = 0;
736 *pcbPostRead = 0;
737 }
738 else
739 {
740 uOffsetInFile = (pImage->pAllocationBitmap[iIndexInAllocationTable] + uSector) * 512;
741 rc = vdIfIoIntFileWriteUser(pImage->pIfIo, pImage->pStorage,
742 uOffsetInFile, pIoCtx, cbToWrite, NULL, NULL);
743 }
744 }
745
746 if (pcbWriteProcess)
747 *pcbWriteProcess = cbToWrite;
748
749 LogFlowFunc(("returns %Rrc\n", rc));
750 return rc;
751}
752
753/** @copydoc VBOXHDDBACKEND::pfnFlush */
754static DECLCALLBACK(int) parallelsFlush(void *pBackendData, PVDIOCTX pIoCtx)
755{
756 int rc = VINF_SUCCESS;
757 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
758
759 LogFlowFunc(("pImage=#%p\n", pImage));
760
761 /* Flush the file, everything is up to date already. */
762 rc = vdIfIoIntFileFlush(pImage->pIfIo, pImage->pStorage, pIoCtx, NULL, NULL);
763
764 LogFlowFunc(("returns %Rrc\n", rc));
765 return rc;
766}
767
768/** @copydoc VBOXHDDBACKEND::pfnGetVersion */
769static DECLCALLBACK(unsigned) parallelsGetVersion(void *pBackendData)
770{
771 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
772 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
773
774 AssertPtr(pImage);
775
776 if (pImage)
777 return PARALLELS_DISK_VERSION;
778 else
779 return 0;
780}
781
782/** @copydoc VBOXHDDBACKEND::pfnGetSectorSize */
783static DECLCALLBACK(uint32_t) parallelsGetSectorSize(void *pBackendData)
784{
785 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
786 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
787 uint32_t cb = 0;
788
789 AssertPtr(pImage);
790
791 if (pImage && pImage->pStorage)
792 cb = 512;
793
794 LogFlowFunc(("returns %llu\n", cb));
795 return cb;
796}
797
798/** @copydoc VBOXHDDBACKEND::pfnGetSize */
799static DECLCALLBACK(uint64_t) parallelsGetSize(void *pBackendData)
800{
801 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
802 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
803 uint64_t cb = 0;
804
805 AssertPtr(pImage);
806
807 if (pImage && pImage->pStorage)
808 cb = pImage->cbSize;
809
810 LogFlowFunc(("returns %llu\n", cb));
811 return cb;
812}
813
814/** @copydoc VBOXHDDBACKEND::pfnGetFileSize */
815static DECLCALLBACK(uint64_t) parallelsGetFileSize(void *pBackendData)
816{
817 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
818 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
819 uint64_t cb = 0;
820
821 AssertPtr(pImage);
822
823 if (pImage && pImage->pStorage)
824 cb = pImage->cbFileCurrent;
825
826 LogFlowFunc(("returns %lld\n", cb));
827 return cb;
828}
829
830/** @copydoc VBOXHDDBACKEND::pfnGetPCHSGeometry */
831static DECLCALLBACK(int) parallelsGetPCHSGeometry(void *pBackendData,
832 PVDGEOMETRY pPCHSGeometry)
833{
834 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p\n", pBackendData, pPCHSGeometry));
835 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
836 int rc;
837
838 AssertPtr(pImage);
839
840 if (pImage)
841 {
842 if (pImage->PCHSGeometry.cCylinders)
843 {
844 *pPCHSGeometry = pImage->PCHSGeometry;
845 rc = VINF_SUCCESS;
846 }
847 else
848 rc = VERR_VD_GEOMETRY_NOT_SET;
849 }
850 else
851 rc = VERR_VD_NOT_OPENED;
852
853 LogFlowFunc(("returns %Rrc (PCHS=%u/%u/%u)\n", rc, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
854 return rc;
855}
856
857/** @copydoc VBOXHDDBACKEND::pfnSetPCHSGeometry */
858static DECLCALLBACK(int) parallelsSetPCHSGeometry(void *pBackendData,
859 PCVDGEOMETRY pPCHSGeometry)
860{
861 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p PCHS=%u/%u/%u\n", pBackendData, pPCHSGeometry, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
862 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
863 int rc;
864
865 AssertPtr(pImage);
866
867 if (pImage)
868 {
869 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
870 {
871 rc = VERR_VD_IMAGE_READ_ONLY;
872 goto out;
873 }
874
875 pImage->PCHSGeometry = *pPCHSGeometry;
876 rc = VINF_SUCCESS;
877 }
878 else
879 rc = VERR_VD_NOT_OPENED;
880
881out:
882 LogFlowFunc(("returns %Rrc\n", rc));
883 return rc;
884}
885
886/** @copydoc VBOXHDDBACKEND::pfnGetLCHSGeometry */
887static DECLCALLBACK(int) parallelsGetLCHSGeometry(void *pBackendData,
888 PVDGEOMETRY pLCHSGeometry)
889{
890 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p\n", pBackendData, pLCHSGeometry));
891 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
892 int rc;
893
894 AssertPtr(pImage);
895
896 if (pImage)
897 {
898 if (pImage->LCHSGeometry.cCylinders)
899 {
900 *pLCHSGeometry = pImage->LCHSGeometry;
901 rc = VINF_SUCCESS;
902 }
903 else
904 rc = VERR_VD_GEOMETRY_NOT_SET;
905 }
906 else
907 rc = VERR_VD_NOT_OPENED;
908
909 LogFlowFunc(("returns %Rrc (LCHS=%u/%u/%u)\n", rc, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
910 return rc;
911}
912
913/** @copydoc VBOXHDDBACKEND::pfnSetLCHSGeometry */
914static DECLCALLBACK(int) parallelsSetLCHSGeometry(void *pBackendData,
915 PCVDGEOMETRY pLCHSGeometry)
916{
917 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p LCHS=%u/%u/%u\n", pBackendData, pLCHSGeometry, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
918 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
919 int rc;
920
921 AssertPtr(pImage);
922
923 if (pImage)
924 {
925 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
926 {
927 rc = VERR_VD_IMAGE_READ_ONLY;
928 goto out;
929 }
930
931 pImage->LCHSGeometry = *pLCHSGeometry;
932 rc = VINF_SUCCESS;
933 }
934 else
935 rc = VERR_VD_NOT_OPENED;
936
937out:
938 LogFlowFunc(("returns %Rrc\n", rc));
939 return rc;
940}
941
942/** @copydoc VBOXHDDBACKEND::pfnGetImageFlags */
943static DECLCALLBACK(unsigned) parallelsGetImageFlags(void *pBackendData)
944{
945 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
946 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
947 unsigned uImageFlags;
948
949 AssertPtr(pImage);
950
951 if (pImage)
952 uImageFlags = pImage->uImageFlags;
953 else
954 uImageFlags = 0;
955
956 LogFlowFunc(("returns %#x\n", uImageFlags));
957 return uImageFlags;
958}
959
960/** @copydoc VBOXHDDBACKEND::pfnGetOpenFlags */
961static DECLCALLBACK(unsigned) parallelsGetOpenFlags(void *pBackendData)
962{
963 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
964 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
965 unsigned uOpenFlags;
966
967 AssertPtr(pImage);
968
969 if (pImage)
970 uOpenFlags = pImage->uOpenFlags;
971 else
972 uOpenFlags = 0;
973
974 LogFlowFunc(("returns %#x\n", uOpenFlags));
975 return uOpenFlags;
976}
977
978/** @copydoc VBOXHDDBACKEND::pfnSetOpenFlags */
979static DECLCALLBACK(int) parallelsSetOpenFlags(void *pBackendData, unsigned uOpenFlags)
980{
981 LogFlowFunc(("pBackendData=%#p\n uOpenFlags=%#x", pBackendData, uOpenFlags));
982 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
983 int rc;
984
985 /* Image must be opened and the new flags must be valid. */
986 if (!pImage || (uOpenFlags & ~( VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO
987 | VD_OPEN_FLAGS_ASYNC_IO | VD_OPEN_FLAGS_SHAREABLE
988 | VD_OPEN_FLAGS_SEQUENTIAL | VD_OPEN_FLAGS_SKIP_CONSISTENCY_CHECKS)))
989 {
990 rc = VERR_INVALID_PARAMETER;
991 goto out;
992 }
993
994 /* Implement this operation via reopening the image. */
995 parallelsFreeImage(pImage, false);
996 rc = parallelsOpenImage(pImage, uOpenFlags);
997
998out:
999 LogFlowFunc(("returns %Rrc\n", rc));
1000 return rc;
1001}
1002
1003/** @copydoc VBOXHDDBACKEND::pfnGetComment */
1004static DECLCALLBACK(int) parallelsGetComment(void *pBackendData, char *pszComment,
1005 size_t cbComment)
1006{
1007 RT_NOREF2(pszComment, cbComment);
1008 LogFlowFunc(("pBackendData=%#p pszComment=%#p cbComment=%zu\n", pBackendData, pszComment, cbComment));
1009 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1010 int rc;
1011
1012 AssertPtr(pImage);
1013
1014 if (pImage)
1015 rc = VERR_NOT_SUPPORTED;
1016 else
1017 rc = VERR_VD_NOT_OPENED;
1018
1019 LogFlowFunc(("returns %Rrc comment='%s'\n", rc, pszComment));
1020 return rc;
1021}
1022
1023/** @copydoc VBOXHDDBACKEND::pfnSetComment */
1024static DECLCALLBACK(int) parallelsSetComment(void *pBackendData, const char *pszComment)
1025{
1026 RT_NOREF1(pszComment);
1027 LogFlowFunc(("pBackendData=%#p pszComment=\"%s\"\n", pBackendData, pszComment));
1028 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1029 int rc;
1030
1031 AssertPtr(pImage);
1032
1033 if (pImage)
1034 {
1035 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1036 rc = VERR_VD_IMAGE_READ_ONLY;
1037 else
1038 rc = VERR_NOT_SUPPORTED;
1039 }
1040 else
1041 rc = VERR_VD_NOT_OPENED;
1042
1043 LogFlowFunc(("returns %Rrc\n", rc));
1044 return rc;
1045}
1046
1047/** @copydoc VBOXHDDBACKEND::pfnGetUuid */
1048static DECLCALLBACK(int) parallelsGetUuid(void *pBackendData, PRTUUID pUuid)
1049{
1050 RT_NOREF1(pUuid);
1051 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1052 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1053 int rc;
1054
1055 AssertPtr(pImage);
1056
1057 if (pImage)
1058 rc = VERR_NOT_SUPPORTED;
1059 else
1060 rc = VERR_VD_NOT_OPENED;
1061
1062 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
1063 return rc;
1064}
1065
1066/** @copydoc VBOXHDDBACKEND::pfnSetUuid */
1067static DECLCALLBACK(int) parallelsSetUuid(void *pBackendData, PCRTUUID pUuid)
1068{
1069 RT_NOREF1(pUuid);
1070 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1071 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1072 int rc;
1073
1074 AssertPtr(pImage);
1075
1076 if (pImage)
1077 {
1078 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
1079 rc = VERR_NOT_SUPPORTED;
1080 else
1081 rc = VERR_VD_IMAGE_READ_ONLY;
1082 }
1083 else
1084 rc = VERR_VD_NOT_OPENED;
1085
1086 LogFlowFunc(("returns %Rrc\n", rc));
1087 return rc;
1088}
1089
1090/** @copydoc VBOXHDDBACKEND::pfnGetModificationUuid */
1091static DECLCALLBACK(int) parallelsGetModificationUuid(void *pBackendData, PRTUUID pUuid)
1092{
1093 RT_NOREF1(pUuid);
1094 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1095 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1096 int rc;
1097
1098 AssertPtr(pImage);
1099
1100 if (pImage)
1101 rc = VERR_NOT_SUPPORTED;
1102 else
1103 rc = VERR_VD_NOT_OPENED;
1104
1105 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
1106 return rc;
1107}
1108
1109/** @copydoc VBOXHDDBACKEND::pfnSetModificationUuid */
1110static DECLCALLBACK(int) parallelsSetModificationUuid(void *pBackendData, PCRTUUID pUuid)
1111{
1112 RT_NOREF1(pUuid);
1113 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1114 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1115 int rc;
1116
1117 AssertPtr(pImage);
1118
1119 if (pImage)
1120 {
1121 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
1122 rc = VERR_NOT_SUPPORTED;
1123 else
1124 rc = VERR_VD_IMAGE_READ_ONLY;
1125 }
1126 else
1127 rc = VERR_VD_NOT_OPENED;
1128
1129 LogFlowFunc(("returns %Rrc\n", rc));
1130 return rc;
1131}
1132
1133/** @copydoc VBOXHDDBACKEND::pfnGetParentUuid */
1134static DECLCALLBACK(int) parallelsGetParentUuid(void *pBackendData, PRTUUID pUuid)
1135{
1136 RT_NOREF1(pUuid);
1137 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1138 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1139 int rc;
1140
1141 AssertPtr(pImage);
1142
1143 if (pImage)
1144 rc = VERR_NOT_SUPPORTED;
1145 else
1146 rc = VERR_VD_NOT_OPENED;
1147
1148 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
1149 return rc;
1150}
1151
1152/** @copydoc VBOXHDDBACKEND::pfnSetParentUuid */
1153static DECLCALLBACK(int) parallelsSetParentUuid(void *pBackendData, PCRTUUID pUuid)
1154{
1155 RT_NOREF1(pUuid);
1156 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1157 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1158 int rc;
1159
1160 AssertPtr(pImage);
1161
1162 if (pImage)
1163 {
1164 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
1165 rc = VERR_NOT_SUPPORTED;
1166 else
1167 rc = VERR_VD_IMAGE_READ_ONLY;
1168 }
1169 else
1170 rc = VERR_VD_NOT_OPENED;
1171
1172 LogFlowFunc(("returns %Rrc\n", rc));
1173 return rc;
1174}
1175
1176/** @copydoc VBOXHDDBACKEND::pfnGetParentModificationUuid */
1177static DECLCALLBACK(int) parallelsGetParentModificationUuid(void *pBackendData, PRTUUID pUuid)
1178{
1179 RT_NOREF1(pUuid);
1180 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1181 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1182 int rc;
1183
1184 AssertPtr(pImage);
1185
1186 if (pImage)
1187 rc = VERR_NOT_SUPPORTED;
1188 else
1189 rc = VERR_VD_NOT_OPENED;
1190
1191 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
1192 return rc;
1193}
1194
1195/** @copydoc VBOXHDDBACKEND::pfnSetParentModificationUuid */
1196static DECLCALLBACK(int) parallelsSetParentModificationUuid(void *pBackendData, PCRTUUID pUuid)
1197{
1198 RT_NOREF1(pUuid);
1199 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1200 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1201 int rc;
1202
1203 AssertPtr(pImage);
1204
1205 if (pImage)
1206 {
1207 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
1208 rc = VERR_NOT_SUPPORTED;
1209 else
1210 rc = VERR_VD_IMAGE_READ_ONLY;
1211 }
1212 else
1213 rc = VERR_VD_NOT_OPENED;
1214
1215 LogFlowFunc(("returns %Rrc\n", rc));
1216 return rc;
1217}
1218
1219/** @copydoc VBOXHDDBACKEND::pfnDump */
1220static DECLCALLBACK(void) parallelsDump(void *pBackendData)
1221{
1222 PPARALLELSIMAGE pImage = (PPARALLELSIMAGE)pBackendData;
1223
1224 AssertPtr(pImage);
1225 if (pImage)
1226 {
1227 vdIfErrorMessage(pImage->pIfError, "Header: Geometry PCHS=%u/%u/%u LCHS=%u/%u/%u\n",
1228 pImage->PCHSGeometry.cCylinders, pImage->PCHSGeometry.cHeads, pImage->PCHSGeometry.cSectors,
1229 pImage->LCHSGeometry.cCylinders, pImage->LCHSGeometry.cHeads, pImage->LCHSGeometry.cSectors);
1230 }
1231}
1232
1233
1234
1235const VBOXHDDBACKEND g_ParallelsBackend =
1236{
1237 /* pszBackendName */
1238 "Parallels",
1239 /* cbSize */
1240 sizeof(VBOXHDDBACKEND),
1241 /* uBackendCaps */
1242 VD_CAP_FILE | VD_CAP_ASYNC | VD_CAP_VFS | VD_CAP_CREATE_DYNAMIC | VD_CAP_DIFF,
1243 /* paFileExtensions */
1244 s_aParallelsFileExtensions,
1245 /* paConfigInfo */
1246 NULL,
1247 /* pfnCheckIfValid */
1248 parallelsCheckIfValid,
1249 /* pfnOpen */
1250 parallelsOpen,
1251 /* pfnCreate */
1252 parallelsCreate,
1253 /* pfnRename */
1254 parallelsRename,
1255 /* pfnClose */
1256 parallelsClose,
1257 /* pfnRead */
1258 parallelsRead,
1259 /* pfnWrite */
1260 parallelsWrite,
1261 /* pfnFlush */
1262 parallelsFlush,
1263 /* pfnDiscard */
1264 NULL,
1265 /* pfnGetVersion */
1266 parallelsGetVersion,
1267 /* pfnGetSectorSize */
1268 parallelsGetSectorSize,
1269 /* pfnGetSize */
1270 parallelsGetSize,
1271 /* pfnGetFileSize */
1272 parallelsGetFileSize,
1273 /* pfnGetPCHSGeometry */
1274 parallelsGetPCHSGeometry,
1275 /* pfnSetPCHSGeometry */
1276 parallelsSetPCHSGeometry,
1277 /* pfnGetLCHSGeometry */
1278 parallelsGetLCHSGeometry,
1279 /* pfnSetLCHSGeometry */
1280 parallelsSetLCHSGeometry,
1281 /* pfnGetImageFlags */
1282 parallelsGetImageFlags,
1283 /* pfnGetOpenFlags */
1284 parallelsGetOpenFlags,
1285 /* pfnSetOpenFlags */
1286 parallelsSetOpenFlags,
1287 /* pfnGetComment */
1288 parallelsGetComment,
1289 /* pfnSetComment */
1290 parallelsSetComment,
1291 /* pfnGetUuid */
1292 parallelsGetUuid,
1293 /* pfnSetUuid */
1294 parallelsSetUuid,
1295 /* pfnGetModificationUuid */
1296 parallelsGetModificationUuid,
1297 /* pfnSetModificationUuid */
1298 parallelsSetModificationUuid,
1299 /* pfnGetParentUuid */
1300 parallelsGetParentUuid,
1301 /* pfnSetParentUuid */
1302 parallelsSetParentUuid,
1303 /* pfnGetParentModificationUuid */
1304 parallelsGetParentModificationUuid,
1305 /* pfnSetParentModificationUuid */
1306 parallelsSetParentModificationUuid,
1307 /* pfnDump */
1308 parallelsDump,
1309 /* pfnGetTimestamp */
1310 NULL,
1311 /* pfnGetParentTimestamp */
1312 NULL,
1313 /* pfnSetParentTimestamp */
1314 NULL,
1315 /* pfnGetParentFilename */
1316 NULL,
1317 /* pfnSetParentFilename */
1318 NULL,
1319 /* pfnComposeLocation */
1320 genericFileComposeLocation,
1321 /* pfnComposeName */
1322 genericFileComposeName,
1323 /* pfnCompact */
1324 NULL,
1325 /* pfnResize */
1326 NULL,
1327 /* pfnRepair */
1328 NULL,
1329 /* pfnTraverseMetadata */
1330 NULL
1331};
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