VirtualBox

source: vbox/trunk/src/VBox/Storage/RAW.cpp@ 38399

Last change on this file since 38399 was 38333, checked in by vboxsync, 13 years ago

Storage/RAW: Remove superfluous line

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 43.0 KB
Line 
1/* $Id: RAW.cpp 38333 2011-08-05 15:37:03Z vboxsync $ */
2/** @file
3 * RawHDDCore - Raw Disk image, Core Code.
4 */
5
6/*
7 * Copyright (C) 2006-2010 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/*******************************************************************************
19* Header Files *
20*******************************************************************************/
21#define LOG_GROUP LOG_GROUP_VD_RAW
22#include <VBox/vd-plugin.h>
23#include <VBox/err.h>
24
25#include <VBox/log.h>
26#include <iprt/assert.h>
27#include <iprt/alloc.h>
28#include <iprt/path.h>
29
30/*******************************************************************************
31* Constants And Macros, Structures and Typedefs *
32*******************************************************************************/
33
34/**
35 * Raw image data structure.
36 */
37typedef struct RAWIMAGE
38{
39 /** Image name. */
40 const char *pszFilename;
41 /** Storage handle. */
42 PVDIOSTORAGE pStorage;
43 /** I/O interface. */
44 PVDINTERFACE pInterfaceIO;
45 /** Async I/O interface callbacks. */
46 PVDINTERFACEIOINT pInterfaceIOCallbacks;
47
48 /** Pointer to the per-disk VD interface list. */
49 PVDINTERFACE pVDIfsDisk;
50 /** Pointer to the per-image VD interface list. */
51 PVDINTERFACE pVDIfsImage;
52
53 /** Error callback. */
54 PVDINTERFACE pInterfaceError;
55 /** Opaque data for error callback. */
56 PVDINTERFACEERROR pInterfaceErrorCallbacks;
57
58 /** Open flags passed by VBoxHD layer. */
59 unsigned uOpenFlags;
60 /** Image flags defined during creation or determined during open. */
61 unsigned uImageFlags;
62 /** Total size of the image. */
63 uint64_t cbSize;
64 /** Position in the image (only truly used for sequential access). */
65 uint64_t offAccess;
66 /** Flag if this is a newly created image. */
67 bool fCreate;
68 /** Physical geometry of this image. */
69 VDGEOMETRY PCHSGeometry;
70 /** Logical geometry of this image. */
71 VDGEOMETRY LCHSGeometry;
72
73} RAWIMAGE, *PRAWIMAGE;
74
75
76/** Size of write operations when filling an image with zeroes. */
77#define RAW_FILL_SIZE (128 * _1K)
78
79/** The maximum reasonable size of a floppy image. */
80#define RAW_MAX_FLOPPY_IMG_SIZE (512 * 82 * 24 * 2)
81
82/*******************************************************************************
83* Static Variables *
84*******************************************************************************/
85
86/** NULL-terminated array of supported file extensions. */
87static const VDFILEEXTENSION s_aRawFileExtensions[] =
88{
89 {"iso", VDTYPE_DVD},
90 {"cdr", VDTYPE_DVD},
91 {"img", VDTYPE_FLOPPY},
92 {"ima", VDTYPE_FLOPPY},
93 {"dsk", VDTYPE_FLOPPY},
94 {"vfd", VDTYPE_FLOPPY},
95 {NULL, VDTYPE_INVALID}
96};
97
98/*******************************************************************************
99* Internal Functions *
100*******************************************************************************/
101
102/**
103 * Internal: signal an error to the frontend.
104 */
105DECLINLINE(int) rawError(PRAWIMAGE pImage, int rc, RT_SRC_POS_DECL,
106 const char *pszFormat, ...)
107{
108 va_list va;
109 va_start(va, pszFormat);
110 if (pImage->pInterfaceError)
111 pImage->pInterfaceErrorCallbacks->pfnError(pImage->pInterfaceError->pvUser, rc, RT_SRC_POS_ARGS,
112 pszFormat, va);
113 va_end(va);
114 return rc;
115}
116
117/**
118 * Internal: signal an informational message to the frontend.
119 */
120DECLINLINE(int) rawMessage(PRAWIMAGE pImage, const char *pszFormat, ...)
121{
122 int rc = VINF_SUCCESS;
123 va_list va;
124 va_start(va, pszFormat);
125 if (pImage->pInterfaceError)
126 rc = pImage->pInterfaceErrorCallbacks->pfnMessage(pImage->pInterfaceError->pvUser,
127 pszFormat, va);
128 va_end(va);
129 return rc;
130}
131
132
133DECLINLINE(int) rawFileOpen(PRAWIMAGE pImage, const char *pszFilename,
134 uint32_t fOpen)
135{
136 return pImage->pInterfaceIOCallbacks->pfnOpen(pImage->pInterfaceIO->pvUser,
137 pszFilename, fOpen,
138 &pImage->pStorage);
139}
140
141DECLINLINE(int) rawFileClose(PRAWIMAGE pImage)
142{
143 return pImage->pInterfaceIOCallbacks->pfnClose(pImage->pInterfaceIO->pvUser,
144 pImage->pStorage);
145}
146
147DECLINLINE(int) rawFileDelete(PRAWIMAGE pImage, const char *pszFilename)
148{
149 return pImage->pInterfaceIOCallbacks->pfnDelete(pImage->pInterfaceIO->pvUser,
150 pszFilename);
151}
152
153DECLINLINE(int) rawFileMove(PRAWIMAGE pImage, const char *pszSrc,
154 const char *pszDst, unsigned fMove)
155{
156 return pImage->pInterfaceIOCallbacks->pfnMove(pImage->pInterfaceIO->pvUser,
157 pszSrc, pszDst, fMove);
158}
159
160DECLINLINE(int) rawFileGetFreeSpace(PRAWIMAGE pImage, const char *pszFilename,
161 int64_t *pcbFree)
162{
163 return pImage->pInterfaceIOCallbacks->pfnGetFreeSpace(pImage->pInterfaceIO->pvUser,
164 pszFilename, pcbFree);
165}
166
167DECLINLINE(int) rawFileGetSize(PRAWIMAGE pImage, uint64_t *pcbSize)
168{
169 return pImage->pInterfaceIOCallbacks->pfnGetSize(pImage->pInterfaceIO->pvUser,
170 pImage->pStorage, pcbSize);
171}
172
173DECLINLINE(int) rawFileSetSize(PRAWIMAGE pImage, uint64_t cbSize)
174{
175 return pImage->pInterfaceIOCallbacks->pfnSetSize(pImage->pInterfaceIO->pvUser,
176 pImage->pStorage, cbSize);
177}
178
179DECLINLINE(int) rawFileWriteSync(PRAWIMAGE pImage, uint64_t uOffset,
180 const void *pvBuffer, size_t cbBuffer,
181 size_t *pcbWritten)
182{
183 return pImage->pInterfaceIOCallbacks->pfnWriteSync(pImage->pInterfaceIO->pvUser,
184 pImage->pStorage, uOffset,
185 pvBuffer, cbBuffer, pcbWritten);
186}
187
188DECLINLINE(int) rawFileReadSync(PRAWIMAGE pImage, uint64_t uOffset,
189 void *pvBuffer, size_t cbBuffer, size_t *pcbRead)
190{
191 return pImage->pInterfaceIOCallbacks->pfnReadSync(pImage->pInterfaceIO->pvUser,
192 pImage->pStorage, uOffset,
193 pvBuffer, cbBuffer, pcbRead);
194}
195
196DECLINLINE(int) rawFileFlushSync(PRAWIMAGE pImage)
197{
198 return pImage->pInterfaceIOCallbacks->pfnFlushSync(pImage->pInterfaceIO->pvUser,
199 pImage->pStorage);
200}
201
202DECLINLINE(int) rawFileReadUserAsync(PRAWIMAGE pImage, uint64_t uOffset,
203 PVDIOCTX pIoCtx, size_t cbRead)
204{
205 return pImage->pInterfaceIOCallbacks->pfnReadUserAsync(pImage->pInterfaceIO->pvUser,
206 pImage->pStorage,
207 uOffset, pIoCtx,
208 cbRead);
209}
210
211DECLINLINE(int) rawFileWriteUserAsync(PRAWIMAGE pImage, uint64_t uOffset,
212 PVDIOCTX pIoCtx, size_t cbWrite,
213 PFNVDXFERCOMPLETED pfnComplete,
214 void *pvCompleteUser)
215{
216 return pImage->pInterfaceIOCallbacks->pfnWriteUserAsync(pImage->pInterfaceIO->pvUser,
217 pImage->pStorage,
218 uOffset, pIoCtx,
219 cbWrite,
220 pfnComplete,
221 pvCompleteUser);
222}
223
224DECLINLINE(int) rawFileFlushAsync(PRAWIMAGE pImage, PVDIOCTX pIoCtx,
225 PFNVDXFERCOMPLETED pfnComplete,
226 void *pvCompleteUser)
227{
228 return pImage->pInterfaceIOCallbacks->pfnFlushAsync(pImage->pInterfaceIO->pvUser,
229 pImage->pStorage,
230 pIoCtx, pfnComplete,
231 pvCompleteUser);
232}
233
234
235/**
236 * Internal. Flush image data to disk.
237 */
238static int rawFlushImage(PRAWIMAGE pImage)
239{
240 int rc = VINF_SUCCESS;
241
242 if ( pImage->pStorage
243 && !(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
244 rc = rawFileFlushSync(pImage);
245
246 return rc;
247}
248
249/**
250 * Internal. Free all allocated space for representing an image except pImage,
251 * and optionally delete the image from disk.
252 */
253static int rawFreeImage(PRAWIMAGE pImage, bool fDelete)
254{
255 int rc = VINF_SUCCESS;
256
257 /* Freeing a never allocated image (e.g. because the open failed) is
258 * not signalled as an error. After all nothing bad happens. */
259 if (pImage)
260 {
261 if (pImage->pStorage)
262 {
263 /* No point updating the file that is deleted anyway. */
264 if (!fDelete)
265 {
266 /* For newly created images in sequential mode fill it to
267 * the nominal size. */
268 if ( pImage->uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL
269 && !(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
270 && pImage->fCreate)
271 {
272 /* Fill rest of image with zeroes, a must for sequential
273 * images to reach the nominal size. */
274 uint64_t uOff;
275 void *pvBuf = RTMemTmpAllocZ(RAW_FILL_SIZE);
276 if (!pvBuf)
277 goto out;
278
279 uOff = pImage->offAccess;
280 /* Write data to all image blocks. */
281 while (uOff < pImage->cbSize)
282 {
283 unsigned cbChunk = (unsigned)RT_MIN(pImage->cbSize,
284 RAW_FILL_SIZE);
285
286 rc = rawFileWriteSync(pImage, uOff, pvBuf, cbChunk,
287 NULL);
288 if (RT_FAILURE(rc))
289 goto out;
290
291 uOff += cbChunk;
292 }
293out:
294 if (pvBuf)
295 RTMemTmpFree(pvBuf);
296 }
297 rawFlushImage(pImage);
298 }
299
300 rawFileClose(pImage);
301 pImage->pStorage = NULL;
302 }
303
304 if (fDelete && pImage->pszFilename)
305 rawFileDelete(pImage, pImage->pszFilename);
306 }
307
308 LogFlowFunc(("returns %Rrc\n", rc));
309 return rc;
310}
311
312/**
313 * Internal: Open an image, constructing all necessary data structures.
314 */
315static int rawOpenImage(PRAWIMAGE pImage, unsigned uOpenFlags)
316{
317 int rc;
318
319 pImage->uOpenFlags = uOpenFlags;
320 pImage->fCreate = false;
321
322 pImage->pInterfaceError = VDInterfaceGet(pImage->pVDIfsDisk, VDINTERFACETYPE_ERROR);
323 if (pImage->pInterfaceError)
324 pImage->pInterfaceErrorCallbacks = VDGetInterfaceError(pImage->pInterfaceError);
325
326 /* Get I/O interface. */
327 pImage->pInterfaceIO = VDInterfaceGet(pImage->pVDIfsImage, VDINTERFACETYPE_IOINT);
328 AssertPtrReturn(pImage->pInterfaceIO, VERR_INVALID_PARAMETER);
329 pImage->pInterfaceIOCallbacks = VDGetInterfaceIOInt(pImage->pInterfaceIO);
330 AssertPtrReturn(pImage->pInterfaceIOCallbacks, VERR_INVALID_PARAMETER);
331
332 /*
333 * Open the image.
334 */
335 rc = rawFileOpen(pImage, pImage->pszFilename,
336 VDOpenFlagsToFileOpenFlags(uOpenFlags,
337 false /* fCreate */));
338 if (RT_FAILURE(rc))
339 {
340 /* Do NOT signal an appropriate error here, as the VD layer has the
341 * choice of retrying the open if it failed. */
342 goto out;
343 }
344
345 rc = rawFileGetSize(pImage, &pImage->cbSize);
346 if (RT_FAILURE(rc))
347 goto out;
348 if (pImage->cbSize % 512)
349 {
350 rc = VERR_VD_RAW_INVALID_HEADER;
351 goto out;
352 }
353 pImage->uImageFlags |= VD_IMAGE_FLAGS_FIXED;
354
355out:
356 if (RT_FAILURE(rc))
357 rawFreeImage(pImage, false);
358 return rc;
359}
360
361/**
362 * Internal: Create a raw image.
363 */
364static int rawCreateImage(PRAWIMAGE pImage, uint64_t cbSize,
365 unsigned uImageFlags, const char *pszComment,
366 PCVDGEOMETRY pPCHSGeometry,
367 PCVDGEOMETRY pLCHSGeometry, unsigned uOpenFlags,
368 PFNVDPROGRESS pfnProgress, void *pvUser,
369 unsigned uPercentStart, unsigned uPercentSpan)
370{
371 int rc;
372 RTFOFF cbFree = 0;
373 uint64_t uOff;
374 void *pvBuf = NULL;
375 int32_t fOpen;
376
377 if (uImageFlags & VD_IMAGE_FLAGS_DIFF)
378 {
379 rc = rawError(pImage, VERR_VD_RAW_INVALID_TYPE, RT_SRC_POS, N_("Raw: cannot create diff image '%s'"), pImage->pszFilename);
380 goto out;
381 }
382 uImageFlags |= VD_IMAGE_FLAGS_FIXED;
383
384 pImage->uOpenFlags = uOpenFlags & ~VD_OPEN_FLAGS_READONLY;
385
386 pImage->uImageFlags = uImageFlags;
387 pImage->fCreate = true;
388 pImage->PCHSGeometry = *pPCHSGeometry;
389 pImage->LCHSGeometry = *pLCHSGeometry;
390
391 pImage->pInterfaceError = VDInterfaceGet(pImage->pVDIfsDisk, VDINTERFACETYPE_ERROR);
392 if (pImage->pInterfaceError)
393 pImage->pInterfaceErrorCallbacks = VDGetInterfaceError(pImage->pInterfaceError);
394
395 /* Get I/O interface. */
396 pImage->pInterfaceIO = VDInterfaceGet(pImage->pVDIfsImage, VDINTERFACETYPE_IOINT);
397 AssertPtrReturn(pImage->pInterfaceIO, VERR_INVALID_PARAMETER);
398 pImage->pInterfaceIOCallbacks = VDGetInterfaceIOInt(pImage->pInterfaceIO);
399 AssertPtrReturn(pImage->pInterfaceIOCallbacks, VERR_INVALID_PARAMETER);
400
401 /* Create image file. */
402 fOpen = VDOpenFlagsToFileOpenFlags(pImage->uOpenFlags, true /* fCreate */);
403 if (uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL)
404 fOpen &= ~RTFILE_O_READ;
405 rc = rawFileOpen(pImage, pImage->pszFilename, fOpen);
406 if (RT_FAILURE(rc))
407 {
408 rc = rawError(pImage, rc, RT_SRC_POS, N_("Raw: cannot create image '%s'"), pImage->pszFilename);
409 goto out;
410 }
411
412 if (!(uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL))
413 {
414 /* Check the free space on the disk and leave early if there is not
415 * sufficient space available. */
416 rc = rawFileGetFreeSpace(pImage, pImage->pszFilename, &cbFree);
417 if (RT_SUCCESS(rc) /* ignore errors */ && ((uint64_t)cbFree < cbSize))
418 {
419 rc = rawError(pImage, VERR_DISK_FULL, RT_SRC_POS, N_("Raw: disk would overflow creating image '%s'"), pImage->pszFilename);
420 goto out;
421 }
422
423 /* Allocate & commit whole file if fixed image, it must be more
424 * effective than expanding file by write operations. */
425 rc = rawFileSetSize(pImage, cbSize);
426 if (RT_FAILURE(rc))
427 {
428 rc = rawError(pImage, rc, RT_SRC_POS, N_("Raw: setting image size failed for '%s'"), pImage->pszFilename);
429 goto out;
430 }
431
432 /* Fill image with zeroes. We do this for every fixed-size image since
433 * on some systems (for example Windows Vista), it takes ages to write
434 * a block near the end of a sparse file and the guest could complain
435 * about an ATA timeout. */
436 pvBuf = RTMemTmpAllocZ(RAW_FILL_SIZE);
437 if (!pvBuf)
438 {
439 rc = VERR_NO_MEMORY;
440 goto out;
441 }
442
443 uOff = 0;
444 /* Write data to all image blocks. */
445 while (uOff < cbSize)
446 {
447 unsigned cbChunk = (unsigned)RT_MIN(cbSize, RAW_FILL_SIZE);
448
449 rc = rawFileWriteSync(pImage, uOff, pvBuf, cbChunk, NULL);
450 if (RT_FAILURE(rc))
451 {
452 rc = rawError(pImage, rc, RT_SRC_POS, N_("Raw: writing block failed for '%s'"), pImage->pszFilename);
453 goto out;
454 }
455
456 uOff += cbChunk;
457
458 if (pfnProgress)
459 {
460 rc = pfnProgress(pvUser,
461 uPercentStart + uOff * uPercentSpan * 98 / (cbSize * 100));
462 if (RT_FAILURE(rc))
463 goto out;
464 }
465 }
466 }
467
468 if (RT_SUCCESS(rc) && pfnProgress)
469 pfnProgress(pvUser, uPercentStart + uPercentSpan * 98 / 100);
470
471 pImage->cbSize = cbSize;
472
473 rc = rawFlushImage(pImage);
474
475out:
476 if (pvBuf)
477 RTMemTmpFree(pvBuf);
478
479 if (RT_SUCCESS(rc) && pfnProgress)
480 pfnProgress(pvUser, uPercentStart + uPercentSpan);
481
482 if (RT_FAILURE(rc))
483 rawFreeImage(pImage, rc != VERR_ALREADY_EXISTS);
484 return rc;
485}
486
487
488/** @copydoc VBOXHDDBACKEND::pfnCheckIfValid */
489static int rawCheckIfValid(const char *pszFilename, PVDINTERFACE pVDIfsDisk,
490 PVDINTERFACE pVDIfsImage, VDTYPE *penmType)
491{
492 LogFlowFunc(("pszFilename=\"%s\" pVDIfsDisk=%#p pVDIfsImage=%#p\n", pszFilename, pVDIfsDisk, pVDIfsImage));
493 PVDIOSTORAGE pStorage = NULL;
494 uint64_t cbFile;
495 int rc = VINF_SUCCESS;
496 char *pszExtension = NULL;
497
498 /* Get I/O interface. */
499 PVDINTERFACE pInterfaceIO = VDInterfaceGet(pVDIfsImage, VDINTERFACETYPE_IOINT);
500 AssertPtrReturn(pInterfaceIO, VERR_INVALID_PARAMETER);
501 PVDINTERFACEIOINT pInterfaceIOCallbacks = VDGetInterfaceIOInt(pInterfaceIO);
502 AssertPtrReturn(pInterfaceIOCallbacks, VERR_INVALID_PARAMETER);
503
504 if ( !VALID_PTR(pszFilename)
505 || !*pszFilename)
506 {
507 rc = VERR_INVALID_PARAMETER;
508 goto out;
509 }
510
511 pszExtension = RTPathExt(pszFilename);
512
513 /*
514 * Open the file and read the footer.
515 */
516 rc = pInterfaceIOCallbacks->pfnOpen(pInterfaceIO->pvUser, pszFilename,
517 VDOpenFlagsToFileOpenFlags(VD_OPEN_FLAGS_READONLY,
518 false /* fCreate */),
519 &pStorage);
520 if (RT_SUCCESS(rc))
521 rc = pInterfaceIOCallbacks->pfnGetSize(pInterfaceIO->pvUser, pStorage,
522 &cbFile);
523
524 /* Try to guess the image type based on the extension. */
525 if ( RT_SUCCESS(rc)
526 && pszExtension)
527 {
528 if ( !RTStrICmp(pszExtension, ".iso")
529 || !RTStrICmp(pszExtension, ".cdr")) /* DVD images. */
530 {
531 /* Note that there are ISO images smaller than 1 MB; it is impossible to distinguish
532 * between raw floppy and CD images based on their size (and cannot be reliably done
533 * based on contents, either).
534 */
535 if (cbFile > 32768 && !(cbFile % 2048))
536 {
537 *penmType = VDTYPE_DVD;
538 rc = VINF_SUCCESS;
539 }
540 else
541 rc = VERR_VD_RAW_INVALID_HEADER;
542 }
543 else if ( !RTStrICmp(pszExtension, ".img")
544 || !RTStrICmp(pszExtension, ".ima")
545 || !RTStrICmp(pszExtension, ".dsk")
546 || !RTStrICmp(pszExtension, ".vfd")) /* Floppy images */
547 {
548 if (!(cbFile % 512) && cbFile <= RAW_MAX_FLOPPY_IMG_SIZE)
549 {
550 *penmType = VDTYPE_FLOPPY;
551 rc = VINF_SUCCESS;
552 }
553 else
554 rc = VERR_VD_RAW_INVALID_HEADER;
555 }
556 else
557 rc = VERR_VD_RAW_INVALID_HEADER;
558 }
559 else
560 rc = VERR_VD_RAW_INVALID_HEADER;
561
562 if (pStorage)
563 pInterfaceIOCallbacks->pfnClose(pInterfaceIO->pvUser, pStorage);
564
565out:
566 LogFlowFunc(("returns %Rrc\n", rc));
567 return rc;
568}
569
570/** @copydoc VBOXHDDBACKEND::pfnOpen */
571static int rawOpen(const char *pszFilename, unsigned uOpenFlags,
572 PVDINTERFACE pVDIfsDisk, PVDINTERFACE pVDIfsImage,
573 VDTYPE enmType, void **ppBackendData)
574{
575 LogFlowFunc(("pszFilename=\"%s\" uOpenFlags=%#x pVDIfsDisk=%#p pVDIfsImage=%#p ppBackendData=%#p\n", pszFilename, uOpenFlags, pVDIfsDisk, pVDIfsImage, ppBackendData));
576 int rc;
577 PRAWIMAGE pImage;
578
579 /* Check open flags. All valid flags are supported. */
580 if (uOpenFlags & ~VD_OPEN_FLAGS_MASK)
581 {
582 rc = VERR_INVALID_PARAMETER;
583 goto out;
584 }
585
586 /* Check remaining arguments. */
587 if ( !VALID_PTR(pszFilename)
588 || !*pszFilename)
589 {
590 rc = VERR_INVALID_PARAMETER;
591 goto out;
592 }
593
594
595 pImage = (PRAWIMAGE)RTMemAllocZ(sizeof(RAWIMAGE));
596 if (!pImage)
597 {
598 rc = VERR_NO_MEMORY;
599 goto out;
600 }
601 pImage->pszFilename = pszFilename;
602 pImage->pStorage = NULL;
603 pImage->pVDIfsDisk = pVDIfsDisk;
604 pImage->pVDIfsImage = pVDIfsImage;
605
606 rc = rawOpenImage(pImage, uOpenFlags);
607 if (RT_SUCCESS(rc))
608 *ppBackendData = pImage;
609 else
610 RTMemFree(pImage);
611
612out:
613 LogFlowFunc(("returns %Rrc (pBackendData=%#p)\n", rc, *ppBackendData));
614 return rc;
615}
616
617/** @copydoc VBOXHDDBACKEND::pfnCreate */
618static int rawCreate(const char *pszFilename, uint64_t cbSize,
619 unsigned uImageFlags, const char *pszComment,
620 PCVDGEOMETRY pPCHSGeometry, PCVDGEOMETRY pLCHSGeometry,
621 PCRTUUID pUuid, unsigned uOpenFlags,
622 unsigned uPercentStart, unsigned uPercentSpan,
623 PVDINTERFACE pVDIfsDisk, PVDINTERFACE pVDIfsImage,
624 PVDINTERFACE pVDIfsOperation, void **ppBackendData)
625{
626 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 ppBackendData=%#p", pszFilename, cbSize, uImageFlags, pszComment, pPCHSGeometry, pLCHSGeometry, pUuid, uOpenFlags, uPercentStart, uPercentSpan, pVDIfsDisk, pVDIfsImage, pVDIfsOperation, ppBackendData));
627 int rc;
628 PRAWIMAGE pImage;
629
630 PFNVDPROGRESS pfnProgress = NULL;
631 void *pvUser = NULL;
632 PVDINTERFACE pIfProgress = VDInterfaceGet(pVDIfsOperation,
633 VDINTERFACETYPE_PROGRESS);
634 PVDINTERFACEPROGRESS pCbProgress = NULL;
635 if (pIfProgress)
636 {
637 pCbProgress = VDGetInterfaceProgress(pIfProgress);
638 if (pCbProgress)
639 pfnProgress = pCbProgress->pfnProgress;
640 pvUser = pIfProgress->pvUser;
641 }
642
643 /* Check open flags. All valid flags are supported. */
644 if (uOpenFlags & ~VD_OPEN_FLAGS_MASK)
645 {
646 rc = VERR_INVALID_PARAMETER;
647 goto out;
648 }
649
650 /* Check remaining arguments. */
651 if ( !VALID_PTR(pszFilename)
652 || !*pszFilename
653 || !VALID_PTR(pPCHSGeometry)
654 || !VALID_PTR(pLCHSGeometry))
655 {
656 rc = VERR_INVALID_PARAMETER;
657 goto out;
658 }
659
660 pImage = (PRAWIMAGE)RTMemAllocZ(sizeof(RAWIMAGE));
661 if (!pImage)
662 {
663 rc = VERR_NO_MEMORY;
664 goto out;
665 }
666 pImage->pszFilename = pszFilename;
667 pImage->pStorage = NULL;
668 pImage->pVDIfsDisk = pVDIfsDisk;
669 pImage->pVDIfsImage = pVDIfsImage;
670
671 rc = rawCreateImage(pImage, cbSize, uImageFlags, pszComment,
672 pPCHSGeometry, pLCHSGeometry, uOpenFlags,
673 pfnProgress, pvUser, uPercentStart, uPercentSpan);
674 if (RT_SUCCESS(rc))
675 {
676 /* So far the image is opened in read/write mode. Make sure the
677 * image is opened in read-only mode if the caller requested that. */
678 if (uOpenFlags & VD_OPEN_FLAGS_READONLY)
679 {
680 rawFreeImage(pImage, false);
681 rc = rawOpenImage(pImage, uOpenFlags);
682 if (RT_FAILURE(rc))
683 {
684 RTMemFree(pImage);
685 goto out;
686 }
687 }
688 *ppBackendData = pImage;
689 }
690 else
691 RTMemFree(pImage);
692
693out:
694 LogFlowFunc(("returns %Rrc (pBackendData=%#p)\n", rc, *ppBackendData));
695 return rc;
696}
697
698/** @copydoc VBOXHDDBACKEND::pfnRename */
699static int rawRename(void *pBackendData, const char *pszFilename)
700{
701 LogFlowFunc(("pBackendData=%#p pszFilename=%#p\n", pBackendData, pszFilename));
702 int rc = VINF_SUCCESS;
703 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
704
705 /* Check arguments. */
706 if ( !pImage
707 || !pszFilename
708 || !*pszFilename)
709 {
710 rc = VERR_INVALID_PARAMETER;
711 goto out;
712 }
713
714 /* Close the image. */
715 rc = rawFreeImage(pImage, false);
716 if (RT_FAILURE(rc))
717 goto out;
718
719 /* Rename the file. */
720 rc = rawFileMove(pImage, pImage->pszFilename, pszFilename, 0);
721 if (RT_FAILURE(rc))
722 {
723 /* The move failed, try to reopen the original image. */
724 int rc2 = rawOpenImage(pImage, pImage->uOpenFlags);
725 if (RT_FAILURE(rc2))
726 rc = rc2;
727
728 goto out;
729 }
730
731 /* Update pImage with the new information. */
732 pImage->pszFilename = pszFilename;
733
734 /* Open the old image with new name. */
735 rc = rawOpenImage(pImage, pImage->uOpenFlags);
736 if (RT_FAILURE(rc))
737 goto out;
738
739out:
740 LogFlowFunc(("returns %Rrc\n", rc));
741 return rc;
742}
743
744/** @copydoc VBOXHDDBACKEND::pfnClose */
745static int rawClose(void *pBackendData, bool fDelete)
746{
747 LogFlowFunc(("pBackendData=%#p fDelete=%d\n", pBackendData, fDelete));
748 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
749 int rc;
750
751 rc = rawFreeImage(pImage, fDelete);
752 RTMemFree(pImage);
753
754 LogFlowFunc(("returns %Rrc\n", rc));
755 return rc;
756}
757
758/** @copydoc VBOXHDDBACKEND::pfnRead */
759static int rawRead(void *pBackendData, uint64_t uOffset, void *pvBuf,
760 size_t cbToRead, size_t *pcbActuallyRead)
761{
762 LogFlowFunc(("pBackendData=%#p uOffset=%llu pvBuf=%#p cbToRead=%zu pcbActuallyRead=%#p\n", pBackendData, uOffset, pvBuf, cbToRead, pcbActuallyRead));
763 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
764 int rc;
765
766 AssertPtr(pImage);
767 Assert(uOffset % 512 == 0);
768 Assert(cbToRead % 512 == 0);
769
770 if ( uOffset + cbToRead > pImage->cbSize
771 || cbToRead == 0)
772 {
773 rc = VERR_INVALID_PARAMETER;
774 goto out;
775 }
776
777 /* For sequential access do not allow to go back. */
778 if ( pImage->uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL
779 && uOffset < pImage->offAccess)
780 {
781 rc = VERR_INVALID_PARAMETER;
782 goto out;
783 }
784
785 rc = rawFileReadSync(pImage, uOffset, pvBuf, cbToRead, NULL);
786 pImage->offAccess = uOffset + cbToRead;
787 if (pcbActuallyRead)
788 *pcbActuallyRead = cbToRead;
789
790out:
791 LogFlowFunc(("returns %Rrc\n", rc));
792 return rc;
793}
794
795/** @copydoc VBOXHDDBACKEND::pfnWrite */
796static int rawWrite(void *pBackendData, uint64_t uOffset, const void *pvBuf,
797 size_t cbToWrite, size_t *pcbWriteProcess,
798 size_t *pcbPreRead, size_t *pcbPostRead, unsigned fWrite)
799{
800 LogFlowFunc(("pBackendData=%#p uOffset=%llu pvBuf=%#p cbToWrite=%zu pcbWriteProcess=%#p pcbPreRead=%#p pcbPostRead=%#p\n", pBackendData, uOffset, pvBuf, cbToWrite, pcbWriteProcess, pcbPreRead, pcbPostRead));
801 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
802 int rc;
803
804 AssertPtr(pImage);
805 Assert(uOffset % 512 == 0);
806 Assert(cbToWrite % 512 == 0);
807
808 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
809 {
810 rc = VERR_VD_IMAGE_READ_ONLY;
811 goto out;
812 }
813
814 if ( uOffset + cbToWrite > pImage->cbSize
815 || cbToWrite == 0)
816 {
817 rc = VERR_INVALID_PARAMETER;
818 goto out;
819 }
820
821 /* For sequential access do not allow to go back. */
822 if ( pImage->uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL
823 && uOffset < pImage->offAccess)
824 {
825 rc = VERR_INVALID_PARAMETER;
826 goto out;
827 }
828
829 rc = rawFileWriteSync(pImage, uOffset, pvBuf, cbToWrite, NULL);
830 pImage->offAccess = uOffset + cbToWrite;
831 if (pcbWriteProcess)
832 *pcbWriteProcess = cbToWrite;
833
834out:
835 LogFlowFunc(("returns %Rrc\n", rc));
836 return rc;
837}
838
839/** @copydoc VBOXHDDBACKEND::pfnFlush */
840static int rawFlush(void *pBackendData)
841{
842 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
843 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
844 int rc;
845
846 rc = rawFlushImage(pImage);
847 LogFlowFunc(("returns %Rrc\n", rc));
848 return rc;
849}
850
851/** @copydoc VBOXHDDBACKEND::pfnGetVersion */
852static unsigned rawGetVersion(void *pBackendData)
853{
854 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
855 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
856
857 AssertPtr(pImage);
858
859 if (pImage)
860 return 1;
861 else
862 return 0;
863}
864
865/** @copydoc VBOXHDDBACKEND::pfnGetSize */
866static uint64_t rawGetSize(void *pBackendData)
867{
868 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
869 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
870 uint64_t cb = 0;
871
872 AssertPtr(pImage);
873
874 if (pImage && pImage->pStorage)
875 cb = pImage->cbSize;
876
877 LogFlowFunc(("returns %llu\n", cb));
878 return cb;
879}
880
881/** @copydoc VBOXHDDBACKEND::pfnGetFileSize */
882static uint64_t rawGetFileSize(void *pBackendData)
883{
884 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
885 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
886 uint64_t cb = 0;
887
888 AssertPtr(pImage);
889
890 if (pImage)
891 {
892 uint64_t cbFile;
893 if (pImage->pStorage)
894 {
895 int rc = rawFileGetSize(pImage, &cbFile);
896 if (RT_SUCCESS(rc))
897 cb += cbFile;
898 }
899 }
900
901 LogFlowFunc(("returns %lld\n", cb));
902 return cb;
903}
904
905/** @copydoc VBOXHDDBACKEND::pfnGetPCHSGeometry */
906static int rawGetPCHSGeometry(void *pBackendData,
907 PVDGEOMETRY pPCHSGeometry)
908{
909 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p\n", pBackendData, pPCHSGeometry));
910 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
911 int rc;
912
913 AssertPtr(pImage);
914
915 if (pImage)
916 {
917 if (pImage->PCHSGeometry.cCylinders)
918 {
919 *pPCHSGeometry = pImage->PCHSGeometry;
920 rc = VINF_SUCCESS;
921 }
922 else
923 rc = VERR_VD_GEOMETRY_NOT_SET;
924 }
925 else
926 rc = VERR_VD_NOT_OPENED;
927
928 LogFlowFunc(("returns %Rrc (PCHS=%u/%u/%u)\n", rc, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
929 return rc;
930}
931
932/** @copydoc VBOXHDDBACKEND::pfnSetPCHSGeometry */
933static int rawSetPCHSGeometry(void *pBackendData,
934 PCVDGEOMETRY pPCHSGeometry)
935{
936 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p PCHS=%u/%u/%u\n", pBackendData, pPCHSGeometry, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
937 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
938 int rc;
939
940 AssertPtr(pImage);
941
942 if (pImage)
943 {
944 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
945 {
946 rc = VERR_VD_IMAGE_READ_ONLY;
947 goto out;
948 }
949
950 pImage->PCHSGeometry = *pPCHSGeometry;
951 rc = VINF_SUCCESS;
952 }
953 else
954 rc = VERR_VD_NOT_OPENED;
955
956out:
957 LogFlowFunc(("returns %Rrc\n", rc));
958 return rc;
959}
960
961/** @copydoc VBOXHDDBACKEND::pfnGetLCHSGeometry */
962static int rawGetLCHSGeometry(void *pBackendData,
963 PVDGEOMETRY pLCHSGeometry)
964{
965 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p\n", pBackendData, pLCHSGeometry));
966 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
967 int rc;
968
969 AssertPtr(pImage);
970
971 if (pImage)
972 {
973 if (pImage->LCHSGeometry.cCylinders)
974 {
975 *pLCHSGeometry = pImage->LCHSGeometry;
976 rc = VINF_SUCCESS;
977 }
978 else
979 rc = VERR_VD_GEOMETRY_NOT_SET;
980 }
981 else
982 rc = VERR_VD_NOT_OPENED;
983
984 LogFlowFunc(("returns %Rrc (LCHS=%u/%u/%u)\n", rc, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
985 return rc;
986}
987
988/** @copydoc VBOXHDDBACKEND::pfnSetLCHSGeometry */
989static int rawSetLCHSGeometry(void *pBackendData,
990 PCVDGEOMETRY pLCHSGeometry)
991{
992 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p LCHS=%u/%u/%u\n", pBackendData, pLCHSGeometry, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
993 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
994 int rc;
995
996 AssertPtr(pImage);
997
998 if (pImage)
999 {
1000 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1001 {
1002 rc = VERR_VD_IMAGE_READ_ONLY;
1003 goto out;
1004 }
1005
1006 pImage->LCHSGeometry = *pLCHSGeometry;
1007 rc = VINF_SUCCESS;
1008 }
1009 else
1010 rc = VERR_VD_NOT_OPENED;
1011
1012out:
1013 LogFlowFunc(("returns %Rrc\n", rc));
1014 return rc;
1015}
1016
1017/** @copydoc VBOXHDDBACKEND::pfnGetImageFlags */
1018static unsigned rawGetImageFlags(void *pBackendData)
1019{
1020 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
1021 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1022 unsigned uImageFlags;
1023
1024 AssertPtr(pImage);
1025
1026 if (pImage)
1027 uImageFlags = pImage->uImageFlags;
1028 else
1029 uImageFlags = 0;
1030
1031 LogFlowFunc(("returns %#x\n", uImageFlags));
1032 return uImageFlags;
1033}
1034
1035/** @copydoc VBOXHDDBACKEND::pfnGetOpenFlags */
1036static unsigned rawGetOpenFlags(void *pBackendData)
1037{
1038 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
1039 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1040 unsigned uOpenFlags;
1041
1042 AssertPtr(pImage);
1043
1044 if (pImage)
1045 uOpenFlags = pImage->uOpenFlags;
1046 else
1047 uOpenFlags = 0;
1048
1049 LogFlowFunc(("returns %#x\n", uOpenFlags));
1050 return uOpenFlags;
1051}
1052
1053/** @copydoc VBOXHDDBACKEND::pfnSetOpenFlags */
1054static int rawSetOpenFlags(void *pBackendData, unsigned uOpenFlags)
1055{
1056 LogFlowFunc(("pBackendData=%#p\n uOpenFlags=%#x", pBackendData, uOpenFlags));
1057 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1058 int rc;
1059
1060 /* Image must be opened and the new flags must be valid. */
1061 if (!pImage || (uOpenFlags & ~(VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO | VD_OPEN_FLAGS_ASYNC_IO | VD_OPEN_FLAGS_SHAREABLE | VD_OPEN_FLAGS_SEQUENTIAL)))
1062 {
1063 rc = VERR_INVALID_PARAMETER;
1064 goto out;
1065 }
1066
1067 /* Implement this operation via reopening the image. */
1068 rc = rawFreeImage(pImage, false);
1069 if (RT_FAILURE(rc))
1070 goto out;
1071 rc = rawOpenImage(pImage, uOpenFlags);
1072
1073out:
1074 LogFlowFunc(("returns %Rrc\n", rc));
1075 return rc;
1076}
1077
1078/** @copydoc VBOXHDDBACKEND::pfnGetComment */
1079static int rawGetComment(void *pBackendData, char *pszComment,
1080 size_t cbComment)
1081{
1082 LogFlowFunc(("pBackendData=%#p pszComment=%#p cbComment=%zu\n", pBackendData, pszComment, cbComment));
1083 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1084 int rc;
1085
1086 AssertPtr(pImage);
1087
1088 if (pImage)
1089 rc = VERR_NOT_SUPPORTED;
1090 else
1091 rc = VERR_VD_NOT_OPENED;
1092
1093 LogFlowFunc(("returns %Rrc comment='%s'\n", rc, pszComment));
1094 return rc;
1095}
1096
1097/** @copydoc VBOXHDDBACKEND::pfnSetComment */
1098static int rawSetComment(void *pBackendData, const char *pszComment)
1099{
1100 LogFlowFunc(("pBackendData=%#p pszComment=\"%s\"\n", pBackendData, pszComment));
1101 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1102 int rc;
1103
1104 AssertPtr(pImage);
1105
1106 if (pImage)
1107 {
1108 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
1109 rc = VERR_VD_IMAGE_READ_ONLY;
1110 else
1111 rc = VERR_NOT_SUPPORTED;
1112 }
1113 else
1114 rc = VERR_VD_NOT_OPENED;
1115
1116 LogFlowFunc(("returns %Rrc\n", rc));
1117 return rc;
1118}
1119
1120/** @copydoc VBOXHDDBACKEND::pfnGetUuid */
1121static int rawGetUuid(void *pBackendData, PRTUUID pUuid)
1122{
1123 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1124 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1125 int rc;
1126
1127 AssertPtr(pImage);
1128
1129 if (pImage)
1130 rc = VERR_NOT_SUPPORTED;
1131 else
1132 rc = VERR_VD_NOT_OPENED;
1133
1134 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
1135 return rc;
1136}
1137
1138/** @copydoc VBOXHDDBACKEND::pfnSetUuid */
1139static int rawSetUuid(void *pBackendData, PCRTUUID pUuid)
1140{
1141 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1142 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1143 int rc;
1144
1145 LogFlowFunc(("%RTuuid\n", pUuid));
1146 AssertPtr(pImage);
1147
1148 if (pImage)
1149 {
1150 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
1151 rc = VERR_NOT_SUPPORTED;
1152 else
1153 rc = VERR_VD_IMAGE_READ_ONLY;
1154 }
1155 else
1156 rc = VERR_VD_NOT_OPENED;
1157
1158 LogFlowFunc(("returns %Rrc\n", rc));
1159 return rc;
1160}
1161
1162/** @copydoc VBOXHDDBACKEND::pfnGetModificationUuid */
1163static int rawGetModificationUuid(void *pBackendData, PRTUUID pUuid)
1164{
1165 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1166 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1167 int rc;
1168
1169 AssertPtr(pImage);
1170
1171 if (pImage)
1172 rc = VERR_NOT_SUPPORTED;
1173 else
1174 rc = VERR_VD_NOT_OPENED;
1175
1176 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
1177 return rc;
1178}
1179
1180/** @copydoc VBOXHDDBACKEND::pfnSetModificationUuid */
1181static int rawSetModificationUuid(void *pBackendData, PCRTUUID pUuid)
1182{
1183 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1184 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1185 int rc;
1186
1187 AssertPtr(pImage);
1188
1189 if (pImage)
1190 {
1191 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
1192 rc = VERR_NOT_SUPPORTED;
1193 else
1194 rc = VERR_VD_IMAGE_READ_ONLY;
1195 }
1196 else
1197 rc = VERR_VD_NOT_OPENED;
1198
1199 LogFlowFunc(("returns %Rrc\n", rc));
1200 return rc;
1201}
1202
1203/** @copydoc VBOXHDDBACKEND::pfnGetParentUuid */
1204static int rawGetParentUuid(void *pBackendData, PRTUUID pUuid)
1205{
1206 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1207 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1208 int rc;
1209
1210 AssertPtr(pImage);
1211
1212 if (pImage)
1213 rc = VERR_NOT_SUPPORTED;
1214 else
1215 rc = VERR_VD_NOT_OPENED;
1216
1217 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
1218 return rc;
1219}
1220
1221/** @copydoc VBOXHDDBACKEND::pfnSetParentUuid */
1222static int rawSetParentUuid(void *pBackendData, PCRTUUID pUuid)
1223{
1224 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1225 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1226 int rc;
1227
1228 AssertPtr(pImage);
1229
1230 if (pImage)
1231 {
1232 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
1233 rc = VERR_NOT_SUPPORTED;
1234 else
1235 rc = VERR_VD_IMAGE_READ_ONLY;
1236 }
1237 else
1238 rc = VERR_VD_NOT_OPENED;
1239
1240 LogFlowFunc(("returns %Rrc\n", rc));
1241 return rc;
1242}
1243
1244/** @copydoc VBOXHDDBACKEND::pfnGetParentModificationUuid */
1245static int rawGetParentModificationUuid(void *pBackendData, PRTUUID pUuid)
1246{
1247 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
1248 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1249 int rc;
1250
1251 AssertPtr(pImage);
1252
1253 if (pImage)
1254 rc = VERR_NOT_SUPPORTED;
1255 else
1256 rc = VERR_VD_NOT_OPENED;
1257
1258 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
1259 return rc;
1260}
1261
1262/** @copydoc VBOXHDDBACKEND::pfnSetParentModificationUuid */
1263static int rawSetParentModificationUuid(void *pBackendData, PCRTUUID pUuid)
1264{
1265 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
1266 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1267 int rc;
1268
1269 AssertPtr(pImage);
1270
1271 if (pImage)
1272 {
1273 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
1274 rc = VERR_NOT_SUPPORTED;
1275 else
1276 rc = VERR_VD_IMAGE_READ_ONLY;
1277 }
1278 else
1279 rc = VERR_VD_NOT_OPENED;
1280
1281 LogFlowFunc(("returns %Rrc\n", rc));
1282 return rc;
1283}
1284
1285/** @copydoc VBOXHDDBACKEND::pfnDump */
1286static void rawDump(void *pBackendData)
1287{
1288 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1289
1290 AssertPtr(pImage);
1291 if (pImage)
1292 {
1293 rawMessage(pImage, "Header: Geometry PCHS=%u/%u/%u LCHS=%u/%u/%u cbSector=%llu\n",
1294 pImage->PCHSGeometry.cCylinders, pImage->PCHSGeometry.cHeads, pImage->PCHSGeometry.cSectors,
1295 pImage->LCHSGeometry.cCylinders, pImage->LCHSGeometry.cHeads, pImage->LCHSGeometry.cSectors,
1296 pImage->cbSize / 512);
1297 }
1298}
1299
1300/** @copydoc VBOXHDDBACKEND::pfnAsyncRead */
1301static int rawAsyncRead(void *pBackendData, uint64_t uOffset, size_t cbRead,
1302 PVDIOCTX pIoCtx, size_t *pcbActuallyRead)
1303{
1304 int rc = VINF_SUCCESS;
1305 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1306
1307 rc = rawFileReadUserAsync(pImage, uOffset, pIoCtx, cbRead);
1308 if (RT_SUCCESS(rc))
1309 *pcbActuallyRead = cbRead;
1310
1311 return rc;
1312}
1313
1314/** @copydoc VBOXHDDBACKEND::pfnAsyncWrite */
1315static int rawAsyncWrite(void *pBackendData, uint64_t uOffset, size_t cbWrite,
1316 PVDIOCTX pIoCtx,
1317 size_t *pcbWriteProcess, size_t *pcbPreRead,
1318 size_t *pcbPostRead, unsigned fWrite)
1319{
1320 int rc = VINF_SUCCESS;
1321 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1322
1323 rc = rawFileWriteUserAsync(pImage, uOffset, pIoCtx, cbWrite, NULL, NULL);
1324 if (RT_SUCCESS(rc))
1325 {
1326 *pcbWriteProcess = cbWrite;
1327 *pcbPostRead = 0;
1328 *pcbPreRead = 0;
1329 }
1330
1331 return rc;
1332}
1333
1334/** @copydoc VBOXHDDBACKEND::pfnAsyncFlush */
1335static int rawAsyncFlush(void *pBackendData, PVDIOCTX pIoCtx)
1336{
1337 int rc = VINF_SUCCESS;
1338 PRAWIMAGE pImage = (PRAWIMAGE)pBackendData;
1339
1340 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
1341 rc = rawFileFlushAsync(pImage, pIoCtx, NULL, NULL);
1342
1343 return rc;
1344}
1345
1346
1347VBOXHDDBACKEND g_RawBackend =
1348{
1349 /* pszBackendName */
1350 "RAW",
1351 /* cbSize */
1352 sizeof(VBOXHDDBACKEND),
1353 /* uBackendCaps */
1354 VD_CAP_CREATE_FIXED | VD_CAP_FILE | VD_CAP_ASYNC | VD_CAP_VFS,
1355 /* paFileExtensions */
1356 s_aRawFileExtensions,
1357 /* paConfigInfo */
1358 NULL,
1359 /* hPlugin */
1360 NIL_RTLDRMOD,
1361 /* pfnCheckIfValid */
1362 rawCheckIfValid,
1363 /* pfnOpen */
1364 rawOpen,
1365 /* pfnCreate */
1366 rawCreate,
1367 /* pfnRename */
1368 rawRename,
1369 /* pfnClose */
1370 rawClose,
1371 /* pfnRead */
1372 rawRead,
1373 /* pfnWrite */
1374 rawWrite,
1375 /* pfnFlush */
1376 rawFlush,
1377 /* pfnGetVersion */
1378 rawGetVersion,
1379 /* pfnGetSize */
1380 rawGetSize,
1381 /* pfnGetFileSize */
1382 rawGetFileSize,
1383 /* pfnGetPCHSGeometry */
1384 rawGetPCHSGeometry,
1385 /* pfnSetPCHSGeometry */
1386 rawSetPCHSGeometry,
1387 /* pfnGetLCHSGeometry */
1388 rawGetLCHSGeometry,
1389 /* pfnSetLCHSGeometry */
1390 rawSetLCHSGeometry,
1391 /* pfnGetImageFlags */
1392 rawGetImageFlags,
1393 /* pfnGetOpenFlags */
1394 rawGetOpenFlags,
1395 /* pfnSetOpenFlags */
1396 rawSetOpenFlags,
1397 /* pfnGetComment */
1398 rawGetComment,
1399 /* pfnSetComment */
1400 rawSetComment,
1401 /* pfnGetUuid */
1402 rawGetUuid,
1403 /* pfnSetUuid */
1404 rawSetUuid,
1405 /* pfnGetModificationUuid */
1406 rawGetModificationUuid,
1407 /* pfnSetModificationUuid */
1408 rawSetModificationUuid,
1409 /* pfnGetParentUuid */
1410 rawGetParentUuid,
1411 /* pfnSetParentUuid */
1412 rawSetParentUuid,
1413 /* pfnGetParentModificationUuid */
1414 rawGetParentModificationUuid,
1415 /* pfnSetParentModificationUuid */
1416 rawSetParentModificationUuid,
1417 /* pfnDump */
1418 rawDump,
1419 /* pfnGetTimeStamp */
1420 NULL,
1421 /* pfnGetParentTimeStamp */
1422 NULL,
1423 /* pfnSetParentTimeStamp */
1424 NULL,
1425 /* pfnGetParentFilename */
1426 NULL,
1427 /* pfnSetParentFilename */
1428 NULL,
1429 /* pfnAsyncRead */
1430 rawAsyncRead,
1431 /* pfnAsyncWrite */
1432 rawAsyncWrite,
1433 /* pfnAsyncFlush */
1434 rawAsyncFlush,
1435 /* pfnComposeLocation */
1436 genericFileComposeLocation,
1437 /* pfnComposeName */
1438 genericFileComposeName,
1439 /* pfnCompact */
1440 NULL,
1441 /* pfnResize */
1442 NULL
1443};
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