VirtualBox

source: vbox/trunk/src/VBox/Storage/VDVfs.cpp@ 70241

Last change on this file since 70241 was 70225, checked in by vboxsync, 7 years ago

VDVfs.cpp: Snapshot chain support.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 21.3 KB
Line 
1/* $Id: VDVfs.cpp 70225 2017-12-19 18:22:08Z vboxsync $ */
2/** @file
3 * Virtual Disk Container implementation. - VFS glue.
4 */
5
6/*
7 * Copyright (C) 2012-2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include <iprt/types.h>
23#include <iprt/assert.h>
24#include <iprt/mem.h>
25#include <iprt/err.h>
26#include <iprt/asm.h>
27#include <iprt/string.h>
28#include <iprt/file.h>
29#include <iprt/sg.h>
30#include <iprt/vfslowlevel.h>
31#include <iprt/poll.h>
32#include <VBox/vd.h>
33
34
35/*********************************************************************************************************************************
36* Structures and Typedefs *
37*********************************************************************************************************************************/
38
39/**
40 * The internal data of a DVM volume I/O stream.
41 */
42typedef struct VDVFSFILE
43{
44 /** The volume the VFS file belongs to. */
45 PVDISK pDisk;
46 /** Current position. */
47 uint64_t offCurPos;
48 /** Flags given during creation. */
49 uint32_t fFlags;
50} VDVFSFILE;
51/** Pointer to a the internal data of a DVM volume file. */
52typedef VDVFSFILE *PVDVFSFILE;
53
54/**
55 * VD read helper taking care of unaligned accesses.
56 *
57 * @return VBox status code.
58 * @param pDisk VD disk container.
59 * @param off Offset to start reading from.
60 * @param pvBuf Pointer to the buffer to read into.
61 * @param cbRead Amount of bytes to read.
62 */
63static int vdReadHelper(PVDISK pDisk, uint64_t off, void *pvBuf, size_t cbRead)
64{
65 int rc = VINF_SUCCESS;
66
67 /* Take shortcut if possible. */
68 if ( off % 512 == 0
69 && cbRead % 512 == 0)
70 rc = VDRead(pDisk, off, pvBuf, cbRead);
71 else
72 {
73 uint8_t *pbBuf = (uint8_t *)pvBuf;
74 uint8_t abBuf[512];
75
76 /* Unaligned access, make it aligned. */
77 if (off % 512 != 0)
78 {
79 uint64_t offAligned = off & ~(uint64_t)(512 - 1);
80 size_t cbToCopy = 512 - (off - offAligned);
81 rc = VDRead(pDisk, offAligned, abBuf, 512);
82 if (RT_SUCCESS(rc))
83 {
84 memcpy(pbBuf, &abBuf[off - offAligned], cbToCopy);
85 pbBuf += cbToCopy;
86 off += cbToCopy;
87 cbRead -= cbToCopy;
88 }
89 }
90
91 if ( RT_SUCCESS(rc)
92 && (cbRead & ~(uint64_t)(512 - 1)))
93 {
94 size_t cbReadAligned = cbRead & ~(uint64_t)(512 - 1);
95
96 Assert(!(off % 512));
97 rc = VDRead(pDisk, off, pbBuf, cbReadAligned);
98 if (RT_SUCCESS(rc))
99 {
100 pbBuf += cbReadAligned;
101 off += cbReadAligned;
102 cbRead -= cbReadAligned;
103 }
104 }
105
106 if ( RT_SUCCESS(rc)
107 && cbRead)
108 {
109 Assert(cbRead < 512);
110 Assert(!(off % 512));
111
112 rc = VDRead(pDisk, off, abBuf, 512);
113 if (RT_SUCCESS(rc))
114 memcpy(pbBuf, abBuf, cbRead);
115 }
116 }
117
118 return rc;
119}
120
121
122/**
123 * VD write helper taking care of unaligned accesses.
124 *
125 * @return VBox status code.
126 * @param pDisk VD disk container.
127 * @param off Offset to start writing to.
128 * @param pvBuf Pointer to the buffer to read from.
129 * @param cbWrite Amount of bytes to write.
130 */
131static int vdWriteHelper(PVDISK pDisk, uint64_t off, const void *pvBuf, size_t cbWrite)
132{
133 int rc = VINF_SUCCESS;
134
135 /* Take shortcut if possible. */
136 if ( off % 512 == 0
137 && cbWrite % 512 == 0)
138 rc = VDWrite(pDisk, off, pvBuf, cbWrite);
139 else
140 {
141 uint8_t *pbBuf = (uint8_t *)pvBuf;
142 uint8_t abBuf[512];
143
144 /* Unaligned access, make it aligned. */
145 if (off % 512 != 0)
146 {
147 uint64_t offAligned = off & ~(uint64_t)(512 - 1);
148 size_t cbToCopy = 512 - (off - offAligned);
149 rc = VDRead(pDisk, offAligned, abBuf, 512);
150 if (RT_SUCCESS(rc))
151 {
152 memcpy(&abBuf[off - offAligned], pbBuf, cbToCopy);
153 rc = VDWrite(pDisk, offAligned, abBuf, 512);
154
155 pbBuf += cbToCopy;
156 off += cbToCopy;
157 cbWrite -= cbToCopy;
158 }
159 }
160
161 if ( RT_SUCCESS(rc)
162 && (cbWrite & ~(uint64_t)(512 - 1)))
163 {
164 size_t cbWriteAligned = cbWrite & ~(uint64_t)(512 - 1);
165
166 Assert(!(off % 512));
167 rc = VDWrite(pDisk, off, pbBuf, cbWriteAligned);
168 if (RT_SUCCESS(rc))
169 {
170 pbBuf += cbWriteAligned;
171 off += cbWriteAligned;
172 cbWrite -= cbWriteAligned;
173 }
174 }
175
176 if ( RT_SUCCESS(rc)
177 && cbWrite)
178 {
179 Assert(cbWrite < 512);
180 Assert(!(off % 512));
181
182 rc = VDRead(pDisk, off, abBuf, 512);
183 if (RT_SUCCESS(rc))
184 {
185 memcpy(abBuf, pbBuf, cbWrite);
186 rc = VDWrite(pDisk, off, abBuf, 512);
187 }
188 }
189 }
190
191 return rc;
192}
193
194
195/**
196 * @interface_method_impl{RTVFSOBJOPS,pfnClose}
197 */
198static DECLCALLBACK(int) vdVfsFile_Close(void *pvThis)
199{
200 PVDVFSFILE pThis = (PVDVFSFILE)pvThis;
201
202 if (pThis->fFlags & VD_VFSFILE_DESTROY_ON_RELEASE)
203 VDDestroy(pThis->pDisk);
204
205 return VINF_SUCCESS;
206}
207
208
209/**
210 * @interface_method_impl{RTVFSOBJOPS,pfnQueryInfo}
211 */
212static DECLCALLBACK(int) vdVfsFile_QueryInfo(void *pvThis, PRTFSOBJINFO pObjInfo, RTFSOBJATTRADD enmAddAttr)
213{
214 PVDVFSFILE pThis = (PVDVFSFILE)pvThis;
215 unsigned const cOpenImages = VDGetCount(pThis->pDisk);
216
217 pObjInfo->cbObject = VDGetSize(pThis->pDisk, cOpenImages - 1);
218 pObjInfo->cbAllocated = 0;
219 for (unsigned iImage = 0; iImage < cOpenImages; iImage++)
220 pObjInfo->cbAllocated += VDGetFileSize(pThis->pDisk, iImage);
221
222 /** @todo enumerate the disk images directly... */
223 RTTimeNow(&pObjInfo->AccessTime);
224 pObjInfo->BirthTime = pObjInfo->AccessTime;
225 pObjInfo->ChangeTime = pObjInfo->AccessTime;
226 pObjInfo->ModificationTime = pObjInfo->AccessTime;
227
228 pObjInfo->Attr.fMode = RTFS_DOS_NT_NORMAL | RTFS_TYPE_FILE | 0644;
229 pObjInfo->Attr.enmAdditional = enmAddAttr;
230 switch (enmAddAttr)
231 {
232 case RTFSOBJATTRADD_UNIX:
233 pObjInfo->Attr.u.Unix.uid = NIL_RTUID;
234 pObjInfo->Attr.u.Unix.gid = NIL_RTGID;
235 pObjInfo->Attr.u.Unix.cHardlinks = 1;
236 pObjInfo->Attr.u.Unix.INodeIdDevice = 0;
237 pObjInfo->Attr.u.Unix.INodeId = 0;
238 pObjInfo->Attr.u.Unix.fFlags = 0;
239 pObjInfo->Attr.u.Unix.GenerationId = 0;
240 pObjInfo->Attr.u.Unix.Device = 0;
241 break;
242
243 case RTFSOBJATTRADD_UNIX_OWNER:
244 pObjInfo->Attr.u.UnixOwner.uid = NIL_RTUID;
245 pObjInfo->Attr.u.UnixOwner.szName[0] = '\0';
246 break;
247 case RTFSOBJATTRADD_UNIX_GROUP:
248 pObjInfo->Attr.u.UnixGroup.gid = NIL_RTGID;
249 pObjInfo->Attr.u.UnixGroup.szName[0] = '\0';
250 break;
251 case RTFSOBJATTRADD_EASIZE:
252 pObjInfo->Attr.u.EASize.cb = 0;
253 break;
254
255 default:
256 AssertFailedReturn(VERR_INVALID_PARAMETER);
257 }
258
259 return VINF_SUCCESS;
260}
261
262
263/**
264 * @interface_method_impl{RTVFSIOSTREAMOPS,pfnRead}
265 */
266static DECLCALLBACK(int) vdVfsFile_Read(void *pvThis, RTFOFF off, PCRTSGBUF pSgBuf, bool fBlocking, size_t *pcbRead)
267{
268 PVDVFSFILE pThis = (PVDVFSFILE)pvThis;
269
270 Assert(pSgBuf->cSegs == 1);
271 NOREF(fBlocking);
272
273 /*
274 * Find the current position and check if it's within the volume.
275 */
276 uint64_t offUnsigned = off < 0 ? pThis->offCurPos : (uint64_t)off;
277 uint64_t const cbImage = VDGetSize(pThis->pDisk, VD_LAST_IMAGE);
278 if (offUnsigned >= cbImage)
279 {
280 if (pcbRead)
281 {
282 *pcbRead = 0;
283 pThis->offCurPos = cbImage;
284 return VINF_EOF;
285 }
286 return VERR_EOF;
287 }
288
289 int rc = VINF_SUCCESS;
290 size_t cbLeftToRead = pSgBuf->paSegs[0].cbSeg;
291 if (offUnsigned + cbLeftToRead <= cbImage)
292 {
293 if (pcbRead)
294 *pcbRead = cbLeftToRead;
295 }
296 else
297 {
298 if (!pcbRead)
299 return VERR_EOF;
300 *pcbRead = cbLeftToRead = (size_t)(cbImage - offUnsigned);
301 rc = VINF_EOF;
302 }
303
304 /*
305 * Ok, we've got a valid stretch within the file. Do the reading.
306 */
307 if (cbLeftToRead > 0)
308 {
309 int rc2 = vdReadHelper(pThis->pDisk, offUnsigned, pSgBuf->paSegs[0].pvSeg, cbLeftToRead);
310 if (RT_SUCCESS(rc2))
311 offUnsigned += cbLeftToRead;
312 else
313 rc = rc2;
314 }
315
316 pThis->offCurPos = offUnsigned;
317 return rc;
318}
319
320
321/**
322 * @interface_method_impl{RTVFSIOSTREAMOPS,pfnWrite}
323 */
324static DECLCALLBACK(int) vdVfsFile_Write(void *pvThis, RTFOFF off, PCRTSGBUF pSgBuf, bool fBlocking, size_t *pcbWritten)
325{
326 PVDVFSFILE pThis = (PVDVFSFILE)pvThis;
327
328 Assert(pSgBuf->cSegs == 1);
329 NOREF(fBlocking);
330
331 /*
332 * Find the current position and check if it's within the volume.
333 * Writing beyond the end of a volume is not supported.
334 */
335 uint64_t offUnsigned = off < 0 ? pThis->offCurPos : (uint64_t)off;
336 uint64_t const cbImage = VDGetSize(pThis->pDisk, VD_LAST_IMAGE);
337 if (offUnsigned >= cbImage)
338 {
339 if (pcbWritten)
340 {
341 *pcbWritten = 0;
342 pThis->offCurPos = cbImage;
343 }
344 return VERR_EOF;
345 }
346
347 size_t cbLeftToWrite;
348 if (offUnsigned + pSgBuf->paSegs[0].cbSeg < cbImage)
349 {
350 cbLeftToWrite = pSgBuf->paSegs[0].cbSeg;
351 if (pcbWritten)
352 *pcbWritten = cbLeftToWrite;
353 }
354 else
355 {
356 if (!pcbWritten)
357 return VERR_EOF;
358 *pcbWritten = cbLeftToWrite = (size_t)(cbImage - offUnsigned);
359 }
360
361 /*
362 * Ok, we've got a valid stretch within the file. Do the reading.
363 */
364 int rc = VINF_SUCCESS;
365 if (cbLeftToWrite > 0)
366 {
367 rc = vdWriteHelper(pThis->pDisk, offUnsigned, pSgBuf->paSegs[0].pvSeg, cbLeftToWrite);
368 if (RT_SUCCESS(rc))
369 offUnsigned += cbLeftToWrite;
370 }
371
372 pThis->offCurPos = offUnsigned;
373 return rc;
374}
375
376
377/**
378 * @interface_method_impl{RTVFSIOSTREAMOPS,pfnFlush}
379 */
380static DECLCALLBACK(int) vdVfsFile_Flush(void *pvThis)
381{
382 PVDVFSFILE pThis = (PVDVFSFILE)pvThis;
383 return VDFlush(pThis->pDisk);
384}
385
386
387/**
388 * @interface_method_impl{RTVFSIOSTREAMOPS,pfnTell}
389 */
390static DECLCALLBACK(int) vdVfsFile_Tell(void *pvThis, PRTFOFF poffActual)
391{
392 PVDVFSFILE pThis = (PVDVFSFILE)pvThis;
393 *poffActual = pThis->offCurPos;
394 return VINF_SUCCESS;
395}
396
397
398/**
399 * @interface_method_impl{RTVFSOBJSETOPS,pfnSetMode}
400 */
401static DECLCALLBACK(int) vdVfsFile_SetMode(void *pvThis, RTFMODE fMode, RTFMODE fMask)
402{
403 NOREF(pvThis);
404 NOREF(fMode);
405 NOREF(fMask);
406 return VERR_NOT_SUPPORTED;
407}
408
409
410/**
411 * @interface_method_impl{RTVFSOBJSETOPS,pfnSetTimes}
412 */
413static DECLCALLBACK(int) vdVfsFile_SetTimes(void *pvThis, PCRTTIMESPEC pAccessTime, PCRTTIMESPEC pModificationTime,
414 PCRTTIMESPEC pChangeTime, PCRTTIMESPEC pBirthTime)
415{
416 NOREF(pvThis);
417 NOREF(pAccessTime);
418 NOREF(pModificationTime);
419 NOREF(pChangeTime);
420 NOREF(pBirthTime);
421 return VERR_NOT_SUPPORTED;
422}
423
424
425/**
426 * @interface_method_impl{RTVFSOBJSETOPS,pfnSetOwner}
427 */
428static DECLCALLBACK(int) vdVfsFile_SetOwner(void *pvThis, RTUID uid, RTGID gid)
429{
430 NOREF(pvThis);
431 NOREF(uid);
432 NOREF(gid);
433 return VERR_NOT_SUPPORTED;
434}
435
436
437/**
438 * @interface_method_impl{RTVFSFILEOPS,pfnSeek}
439 */
440static DECLCALLBACK(int) vdVfsFile_Seek(void *pvThis, RTFOFF offSeek, unsigned uMethod, PRTFOFF poffActual)
441{
442 PVDVFSFILE pThis = (PVDVFSFILE)pvThis;
443
444 /*
445 * Seek relative to which position.
446 */
447 uint64_t offWrt;
448 switch (uMethod)
449 {
450 case RTFILE_SEEK_BEGIN:
451 offWrt = 0;
452 break;
453
454 case RTFILE_SEEK_CURRENT:
455 offWrt = pThis->offCurPos;
456 break;
457
458 case RTFILE_SEEK_END:
459 offWrt = VDGetSize(pThis->pDisk, VD_LAST_IMAGE);
460 break;
461
462 default:
463 return VERR_INTERNAL_ERROR_5;
464 }
465
466 /*
467 * Calc new position, take care to stay without bounds.
468 */
469 uint64_t offNew;
470 if (offSeek == 0)
471 offNew = offWrt;
472 else if (offSeek > 0)
473 {
474 offNew = offWrt + offSeek;
475 if ( offNew < offWrt
476 || offNew > RTFOFF_MAX)
477 offNew = RTFOFF_MAX;
478 }
479 else if ((uint64_t)-offSeek < offWrt)
480 offNew = offWrt + offSeek;
481 else
482 offNew = 0;
483
484 /*
485 * Update the state and set return value.
486 */
487 pThis->offCurPos = offNew;
488
489 *poffActual = offNew;
490 return VINF_SUCCESS;
491}
492
493
494/**
495 * @interface_method_impl{RTVFSFILEOPS,pfnQuerySize}
496 */
497static DECLCALLBACK(int) vdVfsFile_QuerySize(void *pvThis, uint64_t *pcbFile)
498{
499 PVDVFSFILE pThis = (PVDVFSFILE)pvThis;
500 *pcbFile = VDGetSize(pThis->pDisk, VD_LAST_IMAGE);
501 return VINF_SUCCESS;
502}
503
504
505/**
506 * Standard file operations.
507 */
508DECL_HIDDEN_CONST(const RTVFSFILEOPS) g_vdVfsStdFileOps =
509{
510 { /* Stream */
511 { /* Obj */
512 RTVFSOBJOPS_VERSION,
513 RTVFSOBJTYPE_FILE,
514 "VDFile",
515 vdVfsFile_Close,
516 vdVfsFile_QueryInfo,
517 RTVFSOBJOPS_VERSION
518 },
519 RTVFSIOSTREAMOPS_VERSION,
520 RTVFSIOSTREAMOPS_FEAT_NO_SG,
521 vdVfsFile_Read,
522 vdVfsFile_Write,
523 vdVfsFile_Flush,
524 NULL /*PollOne*/,
525 vdVfsFile_Tell,
526 NULL /*Skip*/,
527 NULL /*ZeroFill*/,
528 RTVFSIOSTREAMOPS_VERSION,
529 },
530 RTVFSFILEOPS_VERSION,
531 /*RTVFSIOFILEOPS_FEAT_NO_AT_OFFSET*/ 0,
532 { /* ObjSet */
533 RTVFSOBJSETOPS_VERSION,
534 RT_OFFSETOF(RTVFSFILEOPS, Stream.Obj) - RT_OFFSETOF(RTVFSFILEOPS, ObjSet),
535 vdVfsFile_SetMode,
536 vdVfsFile_SetTimes,
537 vdVfsFile_SetOwner,
538 RTVFSOBJSETOPS_VERSION
539 },
540 vdVfsFile_Seek,
541 vdVfsFile_QuerySize,
542 NULL /*SetSize*/,
543 NULL /*QueryMaxSize*/,
544 RTVFSFILEOPS_VERSION
545};
546
547
548VBOXDDU_DECL(int) VDCreateVfsFileFromDisk(PVDISK pDisk, uint32_t fFlags,
549 PRTVFSFILE phVfsFile)
550{
551 AssertPtrReturn(pDisk, VERR_INVALID_HANDLE);
552 AssertPtrReturn(phVfsFile, VERR_INVALID_POINTER);
553 AssertReturn((fFlags & ~VD_VFSFILE_FLAGS_MASK) == 0, VERR_INVALID_PARAMETER);
554
555 /*
556 * Create the volume file.
557 */
558 RTVFSFILE hVfsFile;
559 PVDVFSFILE pThis;
560 int rc = RTVfsNewFile(&g_vdVfsStdFileOps, sizeof(*pThis), RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_WRITE,
561 NIL_RTVFS, NIL_RTVFSLOCK, &hVfsFile, (void **)&pThis);
562 if (RT_SUCCESS(rc))
563 {
564 pThis->offCurPos = 0;
565 pThis->pDisk = pDisk;
566 pThis->fFlags = fFlags;
567
568 *phVfsFile = hVfsFile;
569 return VINF_SUCCESS;
570 }
571
572 return rc;
573}
574
575
576/**
577 * @interface_method_impl{RTVFSCHAINELEMENTREG,pfnValidate}
578 */
579static DECLCALLBACK(int) vdVfsChain_Validate(PCRTVFSCHAINELEMENTREG pProviderReg, PRTVFSCHAINSPEC pSpec,
580 PRTVFSCHAINELEMSPEC pElement, uint32_t *poffError, PRTERRINFO pErrInfo)
581{
582 RT_NOREF(pProviderReg, pSpec);
583
584 /*
585 * Basic checks.
586 */
587 if (pElement->enmTypeIn != RTVFSOBJTYPE_INVALID)
588 return VERR_VFS_CHAIN_MUST_BE_FIRST_ELEMENT;
589 if ( pElement->enmType != RTVFSOBJTYPE_FILE
590 && pElement->enmType != RTVFSOBJTYPE_IO_STREAM)
591 return VERR_VFS_CHAIN_ONLY_FILE_OR_IOS;
592
593 if (pElement->cArgs < 1)
594 return VERR_VFS_CHAIN_AT_LEAST_ONE_ARG;
595
596 /*
597 * Parse the flag if present, save in pElement->uProvider.
598 */
599 uint32_t fFlags = (pSpec->fOpenFile & RTFILE_O_ACCESS_MASK) == RTFILE_O_READ
600 ? VD_OPEN_FLAGS_READONLY : VD_OPEN_FLAGS_NORMAL;
601 if (pElement->cArgs > 1)
602 {
603 pElement->paArgs[pElement->cArgs - 1].uProvider = true; /* indicates flags */
604 const char *psz = pElement->paArgs[pElement->cArgs - 1].psz;
605 if (*psz)
606 {
607 if ( !strcmp(psz, "ro")
608 || !strcmp(psz, "r"))
609 {
610 fFlags &= ~(VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_NORMAL);
611 fFlags |= VD_OPEN_FLAGS_READONLY;
612 }
613 else if (!strcmp(psz, "rw"))
614 {
615 fFlags &= ~(VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_NORMAL);
616 fFlags |= VD_OPEN_FLAGS_NORMAL;
617 }
618 else if (strlen(psz) <= 4)
619 {
620 *poffError = pElement->paArgs[pElement->cArgs - 1].offSpec;
621 return RTErrInfoSet(pErrInfo, VERR_VFS_CHAIN_INVALID_ARGUMENT, "Expected 'ro' or 'rw' as argument");
622 }
623 else
624 pElement->paArgs[pElement->cArgs - 1].uProvider = false; /* indicates no flags */
625 }
626 }
627
628 pElement->uProvider = fFlags;
629 if ( pElement->cArgs > 2
630 || (pElement->cArgs == 2 && !pElement->paArgs[pElement->cArgs - 1].uProvider))
631 pElement->uProvider |= RT_BIT_64(63);
632 return VINF_SUCCESS;
633}
634
635
636/**
637 * @interface_method_impl{RTVFSCHAINELEMENTREG,pfnInstantiate}
638 */
639static DECLCALLBACK(int) vdVfsChain_Instantiate(PCRTVFSCHAINELEMENTREG pProviderReg, PCRTVFSCHAINSPEC pSpec,
640 PCRTVFSCHAINELEMSPEC pElement, RTVFSOBJ hPrevVfsObj,
641 PRTVFSOBJ phVfsObj, uint32_t *poffError, PRTERRINFO pErrInfo)
642{
643 RT_NOREF(pProviderReg, pSpec, poffError, pErrInfo);
644 AssertReturn(hPrevVfsObj == NIL_RTVFSOBJ, VERR_VFS_CHAIN_IPE);
645
646 /* Determin the format. */
647 char *pszFormat = NULL;
648 VDTYPE enmType = VDTYPE_INVALID;
649 int rc = VDGetFormat(NULL, NULL, pElement->paArgs[0].psz, &pszFormat, &enmType);
650 if (RT_SUCCESS(rc))
651 {
652 PVDISK pDisk = NULL;
653 rc = VDCreate(NULL, enmType, &pDisk);
654 if (RT_SUCCESS(rc))
655 {
656 if (!(pElement->uProvider & RT_BIT_64(63)))
657 rc = VDOpen(pDisk, pszFormat, pElement->paArgs[0].psz, (uint32_t)pElement->uProvider, NULL);
658 else
659 {
660 uint32_t cChain = pElement->cArgs;
661 if (pElement->cArgs == 2 && pElement->paArgs[pElement->cArgs - 1].uProvider != 0)
662 cChain--;
663 uint32_t const fFinal = (uint32_t)pElement->uProvider;
664 uint32_t const fReadOnly = (fFinal & ~(VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_NORMAL)) | VD_OPEN_FLAGS_READONLY;
665 uint32_t iChain;
666 for (iChain = 0; iChain < cChain && RT_SUCCESS(rc); iChain++)
667 rc = VDOpen(pDisk, pszFormat, pElement->paArgs[iChain].psz, iChain + 1 >= cChain ? fFinal : fReadOnly, NULL);
668 }
669 if (RT_SUCCESS(rc))
670 {
671 RTVFSFILE hVfsFile;
672 rc = VDCreateVfsFileFromDisk(pDisk, VD_VFSFILE_DESTROY_ON_RELEASE, &hVfsFile);
673 if (RT_SUCCESS(rc))
674 {
675 RTStrFree(pszFormat);
676
677 *phVfsObj = RTVfsObjFromFile(hVfsFile);
678 RTVfsFileRelease(hVfsFile);
679
680 if (*phVfsObj != NIL_RTVFSOBJ)
681 return VINF_SUCCESS;
682 return VERR_VFS_CHAIN_CAST_FAILED;
683 }
684 }
685 VDDestroy(pDisk);
686 }
687 RTStrFree(pszFormat);
688 }
689 return rc;
690}
691
692
693/**
694 * @interface_method_impl{RTVFSCHAINELEMENTREG,pfnCanReuseElement}
695 */
696static DECLCALLBACK(bool) vdVfsChain_CanReuseElement(PCRTVFSCHAINELEMENTREG pProviderReg,
697 PCRTVFSCHAINSPEC pSpec, PCRTVFSCHAINELEMSPEC pElement,
698 PCRTVFSCHAINSPEC pReuseSpec, PCRTVFSCHAINELEMSPEC pReuseElement)
699{
700 RT_NOREF(pProviderReg, pSpec, pElement, pReuseSpec, pReuseElement);
701 return false;
702}
703
704
705/** VFS chain element 'file'. */
706static RTVFSCHAINELEMENTREG g_rtVfsChainIsoFsVolReg =
707{
708 /* uVersion = */ RTVFSCHAINELEMENTREG_VERSION,
709 /* fReserved = */ 0,
710 /* pszName = */ "vd",
711 /* ListEntry = */ { NULL, NULL },
712 /* pszHelp = */ "Opens a container image using the VD API.\n"
713 "To open a snapshot chain, start with the root image and end with the more recent diff image.\n"
714 "The final argument can be a flag 'ro' or 'r' for read-only, 'rw' for read-write.",
715 /* pfnValidate = */ vdVfsChain_Validate,
716 /* pfnInstantiate = */ vdVfsChain_Instantiate,
717 /* pfnCanReuseElement = */ vdVfsChain_CanReuseElement,
718 /* uEndMarker = */ RTVFSCHAINELEMENTREG_VERSION
719};
720
721RTVFSCHAIN_AUTO_REGISTER_ELEMENT_PROVIDER(&g_rtVfsChainIsoFsVolReg, rtVfsChainIsoFsVolReg);
722
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