VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/dvm/dvm.cpp@ 66299

Last change on this file since 66299 was 64766, checked in by vboxsync, 8 years ago

src/VBox: Make the use of the iterator for RTListForEach()/RTListForEachSafe() more obvious. There is no need to initialize the iterator and we also must not depend on the iterator being NULL if the list was empty.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 21.4 KB
Line 
1/* $Id: dvm.cpp 64766 2016-11-30 10:59:48Z vboxsync $ */
2/** @file
3 * IPRT Disk Volume Management API (DVM) - generic code.
4 */
5
6/*
7 * Copyright (C) 2011-2016 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 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#include <iprt/types.h>
32#include <iprt/assert.h>
33#include <iprt/mem.h>
34#include <iprt/dvm.h>
35#include <iprt/err.h>
36#include <iprt/asm.h>
37#include <iprt/string.h>
38#include <iprt/list.h>
39#include "internal/dvm.h"
40
41
42/*********************************************************************************************************************************
43* Structures and Typedefs *
44*********************************************************************************************************************************/
45
46/**
47 * The internal volume manager structure.
48 */
49typedef struct RTDVMINTERNAL
50{
51 /** The DVM magic (RTDVM_MAGIC). */
52 uint32_t u32Magic;
53 /** The disk descriptor. */
54 RTDVMDISK DvmDisk;
55 /** Pointer to the backend operations table after a successful probe. */
56 PCRTDVMFMTOPS pDvmFmtOps;
57 /** The format specific volume manager data. */
58 RTDVMFMT hVolMgrFmt;
59 /** Flags passed on manager creation. */
60 uint32_t fFlags;
61 /** Reference counter. */
62 uint32_t volatile cRefs;
63 /** List of recognised volumes (RTDVMVOLUMEINTERNAL). */
64 RTLISTANCHOR VolumeList;
65} RTDVMINTERNAL;
66/** Pointer to an internal volume manager. */
67typedef RTDVMINTERNAL *PRTDVMINTERNAL;
68
69/**
70 * The internal volume structure.
71 */
72typedef struct RTDVMVOLUMEINTERNAL
73{
74 /** The DVM volume magic (RTDVMVOLUME_MAGIC). */
75 uint32_t u32Magic;
76 /** Node for the volume list. */
77 RTLISTNODE VolumeNode;
78 /** Pointer to the owning volume manager. */
79 PRTDVMINTERNAL pVolMgr;
80 /** Format specific volume data. */
81 RTDVMVOLUMEFMT hVolFmt;
82 /** Set block status.callback */
83 PFNDVMVOLUMEQUERYBLOCKSTATUS pfnQueryBlockStatus;
84 /** Opaque user data. */
85 void *pvUser;
86 /** Reference counter. */
87 uint32_t volatile cRefs;
88} RTDVMVOLUMEINTERNAL;
89/** Pointer to an internal volume. */
90typedef RTDVMVOLUMEINTERNAL *PRTDVMVOLUMEINTERNAL;
91
92
93/*********************************************************************************************************************************
94* Global variables *
95*********************************************************************************************************************************/
96extern RTDVMFMTOPS g_rtDvmFmtMbr;
97extern RTDVMFMTOPS g_rtDvmFmtGpt;
98extern RTDVMFMTOPS g_rtDvmFmtBsdLbl;
99
100/**
101 * Supported volume formats.
102 */
103static PCRTDVMFMTOPS g_aDvmFmts[] =
104{
105 &g_rtDvmFmtMbr,
106 &g_rtDvmFmtGpt,
107 &g_rtDvmFmtBsdLbl
108};
109
110/**
111 * Descriptions of the volume types.
112 *
113 * This is indexed by RTDVMVOLTYPE.
114 */
115static const char * g_apcszDvmVolTypes[] =
116{
117 "Invalid",
118 "Unknown",
119 "NTFS",
120 "FAT16",
121 "FAT32",
122 "Linux swap",
123 "Linux native",
124 "Linux LVM",
125 "Linux SoftRaid",
126 "FreeBSD",
127 "NetBSD",
128 "OpenBSD",
129 "Mac OS X HFS or HFS+",
130 "Solaris"
131};
132
133/**
134 * Creates a new volume.
135 *
136 * @returns IPRT status code.
137 * @param pThis The DVM map instance.
138 * @param hVolFmt The format specific volume handle.
139 * @param phVol Where to store the generic volume handle on success.
140 */
141static int rtDvmVolumeCreate(PRTDVMINTERNAL pThis, RTDVMVOLUMEFMT hVolFmt,
142 PRTDVMVOLUME phVol)
143{
144 int rc = VINF_SUCCESS;
145 PRTDVMVOLUMEINTERNAL pVol = NULL;
146
147 pVol = (PRTDVMVOLUMEINTERNAL)RTMemAllocZ(sizeof(RTDVMVOLUMEINTERNAL));
148 if (pVol)
149 {
150 pVol->u32Magic = RTDVMVOLUME_MAGIC;
151 pVol->cRefs = 0;
152 pVol->pVolMgr = pThis;
153 pVol->hVolFmt = hVolFmt;
154
155 *phVol = pVol;
156 }
157 else
158 rc = VERR_NO_MEMORY;
159
160 return rc;
161}
162
163/**
164 * Destroys a volume handle.
165 *
166 * @param pThis The volume to destroy.
167 */
168static void rtDvmVolumeDestroy(PRTDVMVOLUMEINTERNAL pThis)
169{
170 PRTDVMINTERNAL pVolMgr = pThis->pVolMgr;
171
172 AssertPtr(pVolMgr);
173
174 /* Close the volume. */
175 pVolMgr->pDvmFmtOps->pfnVolumeClose(pThis->hVolFmt);
176
177 pThis->u32Magic = RTDVMVOLUME_MAGIC_DEAD;
178 pThis->pVolMgr = NULL;
179 pThis->hVolFmt = NIL_RTDVMVOLUMEFMT;
180 RTMemFree(pThis);
181
182 /* Release the reference of the volume manager. */
183 RTDvmRelease(pVolMgr);
184}
185
186RTDECL(int) RTDvmCreate(PRTDVM phVolMgr, PFNDVMREAD pfnRead,
187 PFNDVMWRITE pfnWrite, uint64_t cbDisk,
188 uint64_t cbSector, uint32_t fFlags, void *pvUser)
189{
190 int rc = VINF_SUCCESS;
191 PRTDVMINTERNAL pThis;
192
193 AssertMsgReturn(!(fFlags & ~DVM_FLAGS_MASK),
194 ("Invalid flags given %#x\n", fFlags),
195 VERR_INVALID_PARAMETER);
196
197 pThis = (PRTDVMINTERNAL)RTMemAllocZ(sizeof(RTDVMINTERNAL));
198 if (pThis)
199 {
200 pThis->u32Magic = RTDVM_MAGIC;
201 pThis->DvmDisk.cbDisk = cbDisk;
202 pThis->DvmDisk.cbSector = cbSector;
203 pThis->DvmDisk.pvUser = pvUser;
204 pThis->DvmDisk.pfnRead = pfnRead;
205 pThis->DvmDisk.pfnWrite = pfnWrite;
206 pThis->pDvmFmtOps = NULL;
207 pThis->hVolMgrFmt = NIL_RTDVMFMT;
208 pThis->fFlags = fFlags;
209 pThis->cRefs = 1;
210 RTListInit(&pThis->VolumeList);
211 *phVolMgr = pThis;
212 }
213 else
214 rc = VERR_NO_MEMORY;
215
216 return rc;
217}
218
219RTDECL(uint32_t) RTDvmRetain(RTDVM hVolMgr)
220{
221 PRTDVMINTERNAL pThis = hVolMgr;
222 AssertPtrReturn(pThis, UINT32_MAX);
223 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, UINT32_MAX);
224
225 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
226 AssertMsg(cRefs > 1 && cRefs < _1M, ("%#x %p\n", cRefs, pThis));
227 return cRefs;
228}
229
230/**
231 * Destroys a volume manager handle.
232 *
233 * @param pThis The volume manager to destroy.
234 */
235static void rtDvmDestroy(PRTDVMINTERNAL pThis)
236{
237 if (pThis->hVolMgrFmt != NIL_RTDVMFMT)
238 {
239 AssertPtr(pThis->pDvmFmtOps);
240
241 /* Let the backend do it's own cleanup first. */
242 pThis->pDvmFmtOps->pfnClose(pThis->hVolMgrFmt);
243 pThis->hVolMgrFmt = NIL_RTDVMFMT;
244 }
245
246 pThis->DvmDisk.cbDisk = 0;
247 pThis->DvmDisk.pvUser = NULL;
248 pThis->DvmDisk.pfnRead = NULL;
249 pThis->DvmDisk.pfnWrite = NULL;
250 pThis->u32Magic = RTDVM_MAGIC_DEAD;
251 RTMemFree(pThis);
252}
253
254RTDECL(uint32_t) RTDvmRelease(RTDVM hVolMgr)
255{
256 PRTDVMINTERNAL pThis = hVolMgr;
257 if (pThis == NIL_RTDVM)
258 return 0;
259 AssertPtrReturn(pThis, UINT32_MAX);
260 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, UINT32_MAX);
261
262 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
263 AssertMsg(cRefs < _1M, ("%#x %p\n", cRefs, pThis));
264 if (cRefs == 0)
265 rtDvmDestroy(pThis);
266 return cRefs;
267}
268
269RTDECL(int) RTDvmMapOpen(RTDVM hVolMgr)
270{
271 int rc = VINF_SUCCESS;
272 uint32_t uScoreMax = RTDVM_MATCH_SCORE_UNSUPPORTED;
273 PCRTDVMFMTOPS pDvmFmtOpsMatch = NULL;
274 PRTDVMINTERNAL pThis = hVolMgr;
275 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
276 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, VERR_INVALID_HANDLE);
277 AssertReturn(pThis->hVolMgrFmt == NIL_RTDVMFMT, VERR_INVALID_HANDLE);
278
279 Assert(!pThis->pDvmFmtOps);
280
281 for (unsigned i = 0; i < RT_ELEMENTS(g_aDvmFmts); i++)
282 {
283 uint32_t uScore;
284 PCRTDVMFMTOPS pDvmFmtOps = g_aDvmFmts[i];
285
286 rc = pDvmFmtOps->pfnProbe(&pThis->DvmDisk, &uScore);
287 if ( RT_SUCCESS(rc)
288 && uScore > uScoreMax)
289 {
290 pDvmFmtOpsMatch = pDvmFmtOps;
291 uScoreMax = uScore;
292 }
293 else if (RT_FAILURE(rc))
294 break;
295 }
296
297 if (RT_SUCCESS(rc))
298 {
299 if (uScoreMax > RTDVM_MATCH_SCORE_UNSUPPORTED)
300 {
301 AssertPtr(pDvmFmtOpsMatch);
302
303 /* Open the format. */
304 rc = pDvmFmtOpsMatch->pfnOpen(&pThis->DvmDisk, &pThis->hVolMgrFmt);
305 if (RT_SUCCESS(rc))
306 {
307 uint32_t cVols;
308
309 pThis->pDvmFmtOps = pDvmFmtOpsMatch;
310
311 cVols = pThis->pDvmFmtOps->pfnGetValidVolumes(pThis->hVolMgrFmt);
312
313 /* Construct volume list. */
314 if (cVols)
315 {
316 PRTDVMVOLUMEINTERNAL pVol = NULL;
317 RTDVMVOLUMEFMT hVolFmt = NIL_RTDVMVOLUMEFMT;
318
319 rc = pThis->pDvmFmtOps->pfnQueryFirstVolume(pThis->hVolMgrFmt, &hVolFmt);
320 if (RT_SUCCESS(rc))
321 {
322 rc = rtDvmVolumeCreate(pThis, hVolFmt, &pVol);
323 if (RT_FAILURE(rc))
324 pThis->pDvmFmtOps->pfnVolumeClose(hVolFmt);
325 }
326
327 if (RT_SUCCESS(rc))
328 {
329 cVols--;
330 RTListAppend(&pThis->VolumeList, &pVol->VolumeNode);
331
332 while ( cVols > 0
333 && RT_SUCCESS(rc))
334 {
335 rc = pThis->pDvmFmtOps->pfnQueryNextVolume(pThis->hVolMgrFmt, pVol->hVolFmt, &hVolFmt);
336 if (RT_SUCCESS(rc))
337 {
338 rc = rtDvmVolumeCreate(pThis, hVolFmt, &pVol);
339 if (RT_FAILURE(rc))
340 pThis->pDvmFmtOps->pfnVolumeClose(hVolFmt);
341 else
342 RTListAppend(&pThis->VolumeList, &pVol->VolumeNode);
343 cVols--;
344 }
345 }
346 }
347
348 if (RT_FAILURE(rc))
349 {
350 /* Remove all entries. */
351 PRTDVMVOLUMEINTERNAL pItNext, pIt;
352 RTListForEachSafe(&pThis->VolumeList, pIt, pItNext, RTDVMVOLUMEINTERNAL, VolumeNode)
353 {
354 RTListNodeRemove(&pIt->VolumeNode);
355 rtDvmVolumeDestroy(pIt);
356 }
357 }
358 }
359 }
360 }
361 else
362 rc = VERR_NOT_SUPPORTED;
363 }
364
365 return rc;
366}
367
368RTDECL(int) RTDvmMapInitialize(RTDVM hVolMgr, const char *pszFmt)
369{
370 int rc = VERR_NOT_SUPPORTED;
371 PRTDVMINTERNAL pThis = hVolMgr;
372 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
373 AssertPtrReturn(pszFmt, VERR_INVALID_POINTER);
374 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, VERR_INVALID_HANDLE);
375 AssertReturn(pThis->hVolMgrFmt == NIL_RTDVMFMT, VERR_INVALID_HANDLE);
376
377 for (unsigned i = 0; i < RT_ELEMENTS(g_aDvmFmts); i++)
378 {
379 PCRTDVMFMTOPS pDvmFmtOps = g_aDvmFmts[i];
380
381 if (!RTStrCmp(pDvmFmtOps->pcszFmt, pszFmt))
382 {
383 rc = pDvmFmtOps->pfnInitialize(&pThis->DvmDisk, &pThis->hVolMgrFmt);
384 if (RT_SUCCESS(rc))
385 pThis->pDvmFmtOps = pDvmFmtOps;
386
387 break;
388 }
389 }
390
391 return rc;
392}
393
394RTDECL(const char *) RTDvmMapGetFormat(RTDVM hVolMgr)
395{
396 PRTDVMINTERNAL pThis = hVolMgr;
397 AssertPtrReturn(pThis, NULL);
398 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, NULL);
399 AssertReturn(pThis->hVolMgrFmt != NIL_RTDVMFMT, NULL);
400
401 return pThis->pDvmFmtOps->pcszFmt;
402}
403
404RTDECL(uint32_t) RTDvmMapGetValidVolumes(RTDVM hVolMgr)
405{
406 PRTDVMINTERNAL pThis = hVolMgr;
407 AssertPtrReturn(pThis, UINT32_MAX);
408 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, UINT32_MAX);
409 AssertReturn(pThis->hVolMgrFmt != NIL_RTDVMFMT, UINT32_MAX);
410
411 return pThis->pDvmFmtOps->pfnGetValidVolumes(pThis->hVolMgrFmt);
412}
413
414RTDECL(uint32_t) RTDvmMapGetMaxVolumes(RTDVM hVolMgr)
415{
416 PRTDVMINTERNAL pThis = hVolMgr;
417 AssertPtrReturn(pThis, UINT32_MAX);
418 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, UINT32_MAX);
419 AssertReturn(pThis->hVolMgrFmt != NIL_RTDVMFMT, UINT32_MAX);
420
421 return pThis->pDvmFmtOps->pfnGetMaxVolumes(pThis->hVolMgrFmt);
422}
423
424RTDECL(int) RTDvmMapQueryFirstVolume(RTDVM hVolMgr, PRTDVMVOLUME phVol)
425{
426 int rc = VERR_DVM_MAP_EMPTY;
427 PRTDVMINTERNAL pThis = hVolMgr;
428 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
429 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, VERR_INVALID_HANDLE);
430 AssertReturn(pThis->hVolMgrFmt != NIL_RTDVMFMT, VERR_INVALID_HANDLE);
431 AssertPtrReturn(phVol, VERR_INVALID_POINTER);
432
433 PRTDVMVOLUMEINTERNAL pVol = RTListGetFirst(&pThis->VolumeList, RTDVMVOLUMEINTERNAL, VolumeNode);
434 if (pVol)
435 {
436 rc = VINF_SUCCESS;
437 RTDvmVolumeRetain(pVol);
438 *phVol = pVol;
439 }
440
441 return rc;
442}
443
444RTDECL(int) RTDvmMapQueryNextVolume(RTDVM hVolMgr, RTDVMVOLUME hVol, PRTDVMVOLUME phVolNext)
445{
446 int rc = VERR_DVM_MAP_NO_VOLUME;
447 PRTDVMINTERNAL pThis = hVolMgr;
448 PRTDVMVOLUMEINTERNAL pVol = hVol;
449 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
450 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, VERR_INVALID_HANDLE);
451 AssertReturn(pThis->hVolMgrFmt != NIL_RTDVMFMT, VERR_INVALID_HANDLE);
452 AssertPtrReturn(pVol, VERR_INVALID_HANDLE);
453 AssertReturn(pVol->u32Magic == RTDVMVOLUME_MAGIC, VERR_INVALID_HANDLE);
454 AssertPtrReturn(phVolNext, VERR_INVALID_POINTER);
455
456 PRTDVMVOLUMEINTERNAL pVolNext = RTListGetNext(&pThis->VolumeList, pVol,
457 RTDVMVOLUMEINTERNAL, VolumeNode);
458 if (pVolNext)
459 {
460 rc = VINF_SUCCESS;
461 RTDvmVolumeRetain(pVolNext);
462 *phVolNext = pVolNext;
463 }
464
465 return rc;
466}
467
468RTDECL(int) RTDvmMapQueryBlockStatus(RTDVM hVolMgr, uint64_t off, uint64_t cb,
469 bool *pfAllocated)
470{
471 int rc = VINF_SUCCESS;
472 PRTDVMINTERNAL pThis = hVolMgr;
473 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
474 AssertPtrReturn(pfAllocated, VERR_INVALID_POINTER);
475 AssertReturn(pThis->u32Magic == RTDVM_MAGIC, VERR_INVALID_HANDLE);
476 AssertReturn(pThis->hVolMgrFmt != NIL_RTDVMFMT, VERR_INVALID_HANDLE);
477 AssertReturn(off + cb <= pThis->DvmDisk.cbDisk * pThis->DvmDisk.cbSector,
478 VERR_INVALID_PARAMETER);
479
480 /* Check whether the range is inuse by the volume manager metadata first. */
481 rc = pThis->pDvmFmtOps->pfnQueryRangeUse(pThis->hVolMgrFmt, off, cb, pfAllocated);
482 if (RT_FAILURE(rc))
483 return rc;
484
485 if (!*pfAllocated)
486 {
487 bool fAllocated = false;
488
489 while ( cb > 0
490 && !fAllocated)
491 {
492 bool fVolFound = false;
493 uint64_t cbIntersect;
494 uint64_t offVol;
495
496 /*
497 * Search through all volumes. It is not possible to
498 * get all start sectors and sizes of all volumes here
499 * because volumes can be scattered around the disk for certain formats.
500 * Linux LVM is one example, extents of logical volumes don't need to be
501 * contigous on the medium.
502 */
503 PRTDVMVOLUMEINTERNAL pVol;
504 RTListForEach(&pThis->VolumeList, pVol, RTDVMVOLUMEINTERNAL, VolumeNode)
505 {
506 bool fIntersect = pThis->pDvmFmtOps->pfnVolumeIsRangeIntersecting(pVol->hVolFmt, off,
507 cb, &offVol,
508 &cbIntersect);
509 if (fIntersect)
510 {
511 fVolFound = true;
512 if (pVol->pfnQueryBlockStatus)
513 {
514 bool fVolAllocated = true;
515
516 rc = pVol->pfnQueryBlockStatus(pVol->pvUser, offVol, cbIntersect,
517 &fVolAllocated);
518 if (RT_FAILURE(rc))
519 break;
520 else if (fVolAllocated)
521 {
522 fAllocated = true;
523 break;
524 }
525 }
526 else if (!(pThis->fFlags & DVM_FLAGS_NO_STATUS_CALLBACK_MARK_AS_UNUSED))
527 fAllocated = true;
528 /* else, flag is set, continue. */
529
530 cb -= cbIntersect;
531 off += cbIntersect;
532 break;
533 }
534 }
535
536 if (!fVolFound)
537 {
538 if (pThis->fFlags & DVM_FLAGS_UNUSED_SPACE_MARK_AS_USED)
539 fAllocated = true;
540
541 cb -= pThis->DvmDisk.cbSector;
542 off += pThis->DvmDisk.cbSector;
543 }
544 }
545
546 *pfAllocated = fAllocated;
547 }
548
549 return rc;
550}
551
552RTDECL(uint32_t) RTDvmVolumeRetain(RTDVMVOLUME hVol)
553{
554 PRTDVMVOLUMEINTERNAL pThis = hVol;
555 AssertPtrReturn(pThis, UINT32_MAX);
556 AssertReturn(pThis->u32Magic == RTDVMVOLUME_MAGIC, UINT32_MAX);
557
558 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
559 AssertMsg(cRefs >= 1 && cRefs < _1M, ("%#x %p\n", cRefs, pThis));
560 if (cRefs == 1)
561 RTDvmRetain(pThis->pVolMgr);
562 return cRefs;
563}
564
565RTDECL(uint32_t) RTDvmVolumeRelease(RTDVMVOLUME hVol)
566{
567 PRTDVMVOLUMEINTERNAL pThis = hVol;
568 if (pThis == NIL_RTDVMVOLUME)
569 return 0;
570 AssertPtrReturn(pThis, UINT32_MAX);
571 AssertReturn(pThis->u32Magic == RTDVMVOLUME_MAGIC, UINT32_MAX);
572
573 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
574 AssertMsg(cRefs < _1M, ("%#x %p\n", cRefs, pThis));
575 if (cRefs == 0)
576 {
577 /* Release the volume manager. */
578 pThis->pfnQueryBlockStatus = NULL;
579 RTDvmRelease(pThis->pVolMgr);
580 }
581 return cRefs;
582}
583
584RTDECL(void) RTDvmVolumeSetQueryBlockStatusCallback(RTDVMVOLUME hVol,
585 PFNDVMVOLUMEQUERYBLOCKSTATUS pfnQueryBlockStatus,
586 void *pvUser)
587{
588 PRTDVMVOLUMEINTERNAL pThis = hVol;
589 AssertPtrReturnVoid(pThis);
590 AssertReturnVoid(pThis->u32Magic == RTDVMVOLUME_MAGIC);
591
592 pThis->pfnQueryBlockStatus = pfnQueryBlockStatus;
593 pThis->pvUser = pvUser;
594}
595
596RTDECL(uint64_t) RTDvmVolumeGetSize(RTDVMVOLUME hVol)
597{
598 PRTDVMVOLUMEINTERNAL pThis = hVol;
599 AssertPtrReturn(pThis, 0);
600 AssertReturn(pThis->u32Magic == RTDVMVOLUME_MAGIC, 0);
601
602 return pThis->pVolMgr->pDvmFmtOps->pfnVolumeGetSize(pThis->hVolFmt);
603}
604
605RTDECL(int) RTDvmVolumeQueryName(RTDVMVOLUME hVol, char **ppszVolName)
606{
607 PRTDVMVOLUMEINTERNAL pThis = hVol;
608 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
609 AssertReturn(pThis->u32Magic == RTDVMVOLUME_MAGIC, VERR_INVALID_HANDLE);
610 AssertReturn(ppszVolName, VERR_INVALID_POINTER);
611
612 return pThis->pVolMgr->pDvmFmtOps->pfnVolumeQueryName(pThis->hVolFmt, ppszVolName);
613}
614
615RTDECL(RTDVMVOLTYPE) RTDvmVolumeGetType(RTDVMVOLUME hVol)
616{
617 PRTDVMVOLUMEINTERNAL pThis = hVol;
618 AssertPtrReturn(pThis, RTDVMVOLTYPE_INVALID);
619 AssertReturn(pThis->u32Magic == RTDVMVOLUME_MAGIC, RTDVMVOLTYPE_INVALID);
620
621 return pThis->pVolMgr->pDvmFmtOps->pfnVolumeGetType(pThis->hVolFmt);
622}
623
624RTDECL(uint64_t) RTDvmVolumeGetFlags(RTDVMVOLUME hVol)
625{
626 PRTDVMVOLUMEINTERNAL pThis = hVol;
627 AssertPtrReturn(pThis, UINT64_MAX);
628 AssertReturn(pThis->u32Magic == RTDVMVOLUME_MAGIC, UINT64_MAX);
629
630 return pThis->pVolMgr->pDvmFmtOps->pfnVolumeGetFlags(pThis->hVolFmt);
631}
632
633RTDECL(int) RTDvmVolumeRead(RTDVMVOLUME hVol, uint64_t off, void *pvBuf, size_t cbRead)
634{
635 PRTDVMVOLUMEINTERNAL pThis = hVol;
636 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
637 AssertReturn(pThis->u32Magic == RTDVMVOLUME_MAGIC, VERR_INVALID_HANDLE);
638 AssertReturn(pvBuf, VERR_INVALID_POINTER);
639 AssertReturn(cbRead > 0, VERR_INVALID_PARAMETER);
640
641 return pThis->pVolMgr->pDvmFmtOps->pfnVolumeRead(pThis->hVolFmt, off, pvBuf, cbRead);
642}
643
644RTDECL(int) RTDvmVolumeWrite(RTDVMVOLUME hVol, uint64_t off, const void *pvBuf, size_t cbWrite)
645{
646 PRTDVMVOLUMEINTERNAL pThis = hVol;
647 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
648 AssertReturn(pThis->u32Magic == RTDVMVOLUME_MAGIC, VERR_INVALID_HANDLE);
649 AssertReturn(pvBuf, VERR_INVALID_POINTER);
650 AssertReturn(cbWrite > 0, VERR_INVALID_PARAMETER);
651
652 return pThis->pVolMgr->pDvmFmtOps->pfnVolumeWrite(pThis->hVolFmt, off, pvBuf, cbWrite);
653}
654
655RTDECL(const char *) RTDvmVolumeTypeGetDescr(RTDVMVOLTYPE enmVolType)
656{
657 AssertReturn(enmVolType >= RTDVMVOLTYPE_INVALID && enmVolType < RTDVMVOLTYPE_END, NULL);
658
659 return g_apcszDvmVolTypes[enmVolType];
660}
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