VirtualBox

source: vbox/trunk/src/VBox/Storage/DMG.cpp@ 103131

Last change on this file since 103131 was 98103, checked in by vboxsync, 20 months ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 79.0 KB
RevLine 
[21401]1/* $Id: DMG.cpp 98103 2023-01-17 14:15:46Z vboxsync $ */
2/** @file
[33540]3 * VBoxDMG - Interpreter for Apple Disk Images (DMG).
[21401]4 */
5
6/*
[98103]7 * Copyright (C) 2010-2023 Oracle and/or its affiliates.
[21401]8 *
[96407]9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * SPDX-License-Identifier: GPL-3.0-only
[21401]26 */
27
[57358]28
29/*********************************************************************************************************************************
30* Header Files *
31*********************************************************************************************************************************/
[38623]32#define LOG_GROUP LOG_GROUP_VD_DMG
[33567]33#include <VBox/vd-plugin.h>
[48852]34#include <VBox/vd-ifs.h>
[21401]35#include <VBox/log.h>
36#include <VBox/err.h>
[48852]37
38#include <iprt/asm.h>
39#include <iprt/alloca.h>
[21401]40#include <iprt/assert.h>
[48852]41#include <iprt/base64.h>
42#include <iprt/ctype.h>
[21401]43#include <iprt/mem.h>
44#include <iprt/string.h>
[32768]45#include <iprt/zip.h>
[48852]46#include <iprt/formats/xar.h>
[21401]47
[50988]48#include "VDBackends.h"
[66494]49#include "VDBackendsInline.h"
[48852]50
[57358]51
52/*********************************************************************************************************************************
53* Structures and Typedefs *
54*********************************************************************************************************************************/
[48871]55#if 0
56/** @def VBOX_WITH_DIRECT_XAR_ACCESS
57 * When defined, we will use RTVfs to access the XAR file instead of going
58 * the slightly longer way thru the VFS -> VD wrapper. */
59# define VBOX_WITH_DIRECT_XAR_ACCESS
60#endif
[32432]61
62/** Sector size, multiply with all sector counts to get number of bytes. */
[32768]63#define DMG_SECTOR_SIZE 512
[32432]64
[32768]65/** Convert block number/size to byte offset/size. */
66#define DMG_BLOCK2BYTE(u) ((uint64_t)(u) << 9)
67
68/** Convert byte offset/size to block number/size. */
69#define DMG_BYTE2BLOCK(u) ((u) >> 9)
70
[21401]71/**
72 * UDIF checksum structure.
73 */
[32768]74typedef struct DMGUDIFCKSUM
[21401]75{
76 uint32_t u32Kind; /**< The kind of checksum. */
77 uint32_t cBits; /**< The size of the checksum. */
78 union
79 {
80 uint8_t au8[128]; /**< 8-bit view. */
81 uint32_t au32[32]; /**< 32-bit view. */
82 } uSum; /**< The checksum. */
[32768]83} DMGUDIFCKSUM;
84AssertCompileSize(DMGUDIFCKSUM, 8 + 128);
85typedef DMGUDIFCKSUM *PDMGUDIFCKSUM;
86typedef const DMGUDIFCKSUM *PCDMGUDIFCKSUM;
[21401]87
[32768]88/** @name Checksum Kind (DMGUDIFCKSUM::u32Kind)
[21401]89 * @{ */
90/** No checksum. */
[32768]91#define DMGUDIFCKSUM_NONE UINT32_C(0)
[21401]92/** CRC-32. */
[32768]93#define DMGUDIFCKSUM_CRC32 UINT32_C(2)
[21401]94/** @} */
95
96/**
97 * UDIF ID.
98 * This is kind of like a UUID only it isn't, but we'll use the UUID
99 * representation of it for simplicity.
100 */
[32768]101typedef RTUUID DMGUDIFID;
102AssertCompileSize(DMGUDIFID, 16);
103typedef DMGUDIFID *PDMGUDIFID;
104typedef const DMGUDIFID *PCDMGUDIFID;
[21401]105
106/**
107 * UDIF footer used by Apple Disk Images (DMG).
108 *
109 * This is a footer placed 512 bytes from the end of the file. Typically a DMG
110 * file starts with the data, which is followed by the block table and then ends
111 * with this structure.
112 *
113 * All fields are stored in big endian format.
114 */
115#pragma pack(1)
[32768]116typedef struct DMGUDIF
[21401]117{
[32768]118 uint32_t u32Magic; /**< 0x000 - Magic, 'koly' (DMGUDIF_MAGIC). (fUDIFSignature) */
119 uint32_t u32Version; /**< 0x004 - The UDIF version (DMGUDIF_VER_CURRENT). (fUDIFVersion) */
[21401]120 uint32_t cbFooter; /**< 0x008 - The size of the this structure (512). (fUDIFHeaderSize) */
121 uint32_t fFlags; /**< 0x00c - Flags. (fUDIFFlags) */
122 uint64_t offRunData; /**< 0x010 - Where the running data fork starts (usually 0). (fUDIFRunningDataForkOffset) */
123 uint64_t offData; /**< 0x018 - Where the data fork starts (usually 0). (fUDIFDataForkOffset) */
124 uint64_t cbData; /**< 0x020 - Size of the data fork (in bytes). (fUDIFDataForkLength) */
125 uint64_t offRsrc; /**< 0x028 - Where the resource fork starts (usually cbData or 0). (fUDIFRsrcForkOffset) */
126 uint64_t cbRsrc; /**< 0x030 - The size of the resource fork. (fUDIFRsrcForkLength)*/
127 uint32_t iSegment; /**< 0x038 - The segment number of this file. (fUDIFSegmentNumber) */
128 uint32_t cSegments; /**< 0x03c - The number of segments. (fUDIFSegmentCount) */
[32768]129 DMGUDIFID SegmentId; /**< 0x040 - The segment ID. (fUDIFSegmentID) */
130 DMGUDIFCKSUM DataCkSum; /**< 0x050 - The data checksum. (fUDIFDataForkChecksum) */
[21401]131 uint64_t offXml; /**< 0x0d8 - The XML offset (.plist kind of data). (fUDIFXMLOffset) */
132 uint64_t cbXml; /**< 0x0e0 - The size of the XML. (fUDIFXMLSize) */
133 uint8_t abUnknown[120]; /**< 0x0e8 - Unknown stuff, hdiutil doesn't dump it... */
[32768]134 DMGUDIFCKSUM MasterCkSum; /**< 0x160 - The master checksum. (fUDIFMasterChecksum) */
[21401]135 uint32_t u32Type; /**< 0x1e8 - The image type. (fUDIFImageVariant) */
136 uint64_t cSectors; /**< 0x1ec - The sector count. Warning! Unaligned! (fUDISectorCount) */
137 uint32_t au32Unknown[3]; /**< 0x1f4 - Unknown stuff, hdiutil doesn't dump it... */
[32768]138} DMGUDIF;
[33534]139#pragma pack()
[32768]140AssertCompileSize(DMGUDIF, 512);
141AssertCompileMemberOffset(DMGUDIF, cbRsrc, 0x030);
142AssertCompileMemberOffset(DMGUDIF, cbXml, 0x0e0);
143AssertCompileMemberOffset(DMGUDIF, cSectors, 0x1ec);
[21401]144
[32768]145typedef DMGUDIF *PDMGUDIF;
146typedef const DMGUDIF *PCDMGUDIF;
[21401]147
[32768]148/** The UDIF magic 'koly' (DMGUDIF::u32Magic). */
149#define DMGUDIF_MAGIC UINT32_C(0x6b6f6c79)
[21401]150
[32768]151/** The current UDIF version (DMGUDIF::u32Version).
[21401]152 * This is currently the only we recognizes and will create. */
[32768]153#define DMGUDIF_VER_CURRENT 4
[21401]154
[32768]155/** @name UDIF flags (DMGUDIF::fFlags).
[21401]156 * @{ */
157/** Flatten image whatever that means.
158 * (hdiutil -debug calls it kUDIFFlagsFlattened.) */
[32768]159#define DMGUDIF_FLAGS_FLATTENED RT_BIT_32(0)
[21401]160/** Internet enabled image.
161 * (hdiutil -debug calls it kUDIFFlagsInternetEnabled) */
[32768]162#define DMGUDIF_FLAGS_INET_ENABLED RT_BIT_32(2)
[21401]163/** Mask of known bits. */
[32768]164#define DMGUDIF_FLAGS_KNOWN_MASK (RT_BIT_32(0) | RT_BIT_32(2))
[21401]165/** @} */
166
[32768]167/** @name UDIF Image Types (DMGUDIF::u32Type).
[21401]168 * @{ */
169/** Device image type. (kUDIFDeviceImageType) */
[32768]170#define DMGUDIF_TYPE_DEVICE 1
[21401]171/** Device image type. (kUDIFPartitionImageType) */
[32768]172#define DMGUDIF_TYPE_PARTITION 2
[21401]173/** @} */
174
175/**
[32432]176 * BLKX data.
177 *
[33540]178 * This contains the start offset and size of raw data stored in the image.
[32432]179 *
180 * All fields are stored in big endian format.
181 */
182#pragma pack(1)
[32768]183typedef struct DMGBLKX
[32432]184{
[32768]185 uint32_t u32Magic; /**< 0x000 - Magic, 'mish' (DMGBLKX_MAGIC). */
186 uint32_t u32Version; /**< 0x004 - The BLKX version (DMGBLKX_VER_CURRENT). */
[32432]187 uint64_t cSectornumberFirst; /**< 0x008 - The first sector number the block represents in the virtual device. */
188 uint64_t cSectors; /**< 0x010 - Number of sectors this block represents. */
189 uint64_t offDataStart; /**< 0x018 - Start offset for raw data. */
190 uint32_t cSectorsDecompress; /**< 0x020 - Size of the buffer in sectors needed to decompress. */
191 uint32_t u32BlocksDescriptor; /**< 0x024 - Blocks descriptor. */
192 uint8_t abReserved[24];
[32768]193 DMGUDIFCKSUM BlkxCkSum; /**< 0x03c - Checksum for the BLKX table. */
[32432]194 uint32_t cBlocksRunCount; /**< 0x - Number of entries in the blkx run table afterwards. */
[32768]195} DMGBLKX;
[33534]196#pragma pack()
[32768]197AssertCompileSize(DMGBLKX, 204);
[32432]198
[32768]199typedef DMGBLKX *PDMGBLKX;
200typedef const DMGBLKX *PCDMGBLKX;
[32432]201
[32768]202/** The BLKX magic 'mish' (DMGBLKX::u32Magic). */
203#define DMGBLKX_MAGIC UINT32_C(0x6d697368)
204/** BLKX version (DMGBLKX::u32Version). */
205#define DMGBLKX_VERSION UINT32_C(0x00000001)
[32432]206
207/** Blocks descriptor type: entire device. */
[32768]208#define DMGBLKX_DESC_ENTIRE_DEVICE UINT32_C(0xfffffffe)
[32432]209
210/**
211 * BLKX table descriptor.
212 *
213 * All fields are stored in big endian format.
214 */
215#pragma pack(1)
[32768]216typedef struct DMGBLKXDESC
[32432]217{
218 uint32_t u32Type; /**< 0x000 - Type of the descriptor. */
219 uint32_t u32Reserved; /**< 0x004 - Reserved, but contains +beg or +end in case thisi is a comment descriptor. */
220 uint64_t u64SectorStart; /**< 0x008 - First sector number in the block this entry describes. */
221 uint64_t u64SectorCount; /**< 0x010 - Number of sectors this entry describes. */
222 uint64_t offData; /**< 0x018 - Offset in the image where the data starts. */
223 uint64_t cbData; /**< 0x020 - Number of bytes in the image. */
[32768]224} DMGBLKXDESC;
[33534]225#pragma pack()
[32768]226AssertCompileSize(DMGBLKXDESC, 40);
[32432]227
[32768]228typedef DMGBLKXDESC *PDMGBLKXDESC;
229typedef const DMGBLKXDESC *PCDMGBLKXDESC;
[32432]230
231/** Raw image data type. */
[32768]232#define DMGBLKXDESC_TYPE_RAW 1
[32432]233/** Ignore type. */
[32768]234#define DMGBLKXDESC_TYPE_IGNORE 2
235/** Compressed with zlib type. */
236#define DMGBLKXDESC_TYPE_ZLIB UINT32_C(0x80000005)
[32432]237/** Comment type. */
[32768]238#define DMGBLKXDESC_TYPE_COMMENT UINT32_C(0x7ffffffe)
[32432]239/** Terminator type. */
[32768]240#define DMGBLKXDESC_TYPE_TERMINATOR UINT32_C(0xffffffff)
[32432]241
242/**
[21401]243 * UDIF Resource Entry.
244 */
[32768]245typedef struct DMGUDIFRSRCENTRY
[21401]246{
247 /** The ID. */
248 int32_t iId;
249 /** Attributes. */
250 uint32_t fAttributes;
251 /** The name. */
252 char *pszName;
253 /** The CoreFoundation name. Can be NULL. */
254 char *pszCFName;
255 /** The size of the data. */
256 size_t cbData;
257 /** The raw data. */
258 uint8_t *pbData;
[32768]259} DMGUDIFRSRCENTRY;
[21401]260/** Pointer to an UDIF resource entry. */
[32768]261typedef DMGUDIFRSRCENTRY *PDMGUDIFRSRCENTRY;
[21401]262/** Pointer to a const UDIF resource entry. */
[32768]263typedef DMGUDIFRSRCENTRY const *PCDMGUDIFRSRCENTRY;
[21401]264
265/**
266 * UDIF Resource Array.
267 */
[32768]268typedef struct DMGUDIFRSRCARRAY
[21401]269{
270 /** The array name. */
271 char szName[12];
272 /** The number of occupied entries. */
273 uint32_t cEntries;
274 /** The array entries.
275 * A lazy bird ASSUME there are no more than 4 entries in any DMG. Increase the
[32432]276 * size if DMGs with more are found.
277 * r=aeichner: Saw one with 6 here (image of a whole DVD) */
[32768]278 DMGUDIFRSRCENTRY aEntries[10];
279} DMGUDIFRSRCARRAY;
[21401]280/** Pointer to a UDIF resource array. */
[32768]281typedef DMGUDIFRSRCARRAY *PDMGUDIFRSRCARRAY;
[21401]282/** Pointer to a const UDIF resource array. */
[32768]283typedef DMGUDIFRSRCARRAY const *PCDMGUDIFRSRCARRAY;
[21401]284
[32432]285/**
286 * DMG extent types.
287 */
288typedef enum DMGEXTENTTYPE
289{
290 /** Null, never used. */
291 DMGEXTENTTYPE_NULL = 0,
292 /** Raw image data. */
293 DMGEXTENTTYPE_RAW,
294 /** Zero extent, reads return 0 and writes have no effect. */
295 DMGEXTENTTYPE_ZERO,
[32768]296 /** Compressed extent - compression method ZLIB. */
297 DMGEXTENTTYPE_COMP_ZLIB,
[32432]298 /** 32bit hack. */
299 DMGEXTENTTYPE_32BIT_HACK = 0x7fffffff
300} DMGEXTENTTYPE, *PDMGEXTENTTYPE;
[21401]301
[32432]302/**
303 * DMG extent mapping a virtual image block to real file offsets.
304 */
305typedef struct DMGEXTENT
306{
307 /** Extent type. */
308 DMGEXTENTTYPE enmType;
[32768]309 /** First sector this extent describes. */
310 uint64_t uSectorExtent;
311 /** Number of sectors this extent describes. */
312 uint64_t cSectorsExtent;
[32432]313 /** Start offset in the real file. */
314 uint64_t offFileStart;
[32768]315 /** Number of bytes for the extent data in the file. */
316 uint64_t cbFile;
[32432]317} DMGEXTENT;
318/** Pointer to an DMG extent. */
319typedef DMGEXTENT *PDMGEXTENT;
[21401]320
321/**
322 * VirtualBox Apple Disk Image (DMG) interpreter instance data.
323 */
[32432]324typedef struct DMGIMAGE
[21401]325{
[32536]326 /** Image name.
327 * Kept around for logging and delete-on-close purposes. */
328 const char *pszFilename;
[32432]329 /** Storage handle. */
330 PVDIOSTORAGE pStorage;
[32536]331
332 /** Pointer to the per-disk VD interface list. */
333 PVDINTERFACE pVDIfsDisk;
334 /** Pointer to the per-image VD interface list. */
335 PVDINTERFACE pVDIfsImage;
[38469]336 /** Error interface. */
337 PVDINTERFACEERROR pIfError;
[48852]338 /** I/O interface - careful accessing this because of hDmgFileInXar. */
339 PVDINTERFACEIOINT pIfIoXxx;
[32536]340
[48852]341
342 /** The VFS file handle for a DMG within a XAR archive. */
343 RTVFSFILE hDmgFileInXar;
344 /** XAR file system stream handle.
345 * Sitting on this isn't really necessary, but insurance against the XAR code
346 * changes making back references from child objects to the stream itself. */
347 RTVFSFSSTREAM hXarFss;
348
[21401]349 /** Flags the image was opened with. */
[32432]350 uint32_t uOpenFlags;
351 /** Image flags. */
352 unsigned uImageFlags;
[32536]353 /** Total size of the virtual image. */
354 uint64_t cbSize;
355 /** Size of the image. */
356 uint64_t cbFile;
357 /** Physical geometry of this image. */
358 VDGEOMETRY PCHSGeometry;
359 /** Logical geometry of this image. */
360 VDGEOMETRY LCHSGeometry;
361
[21401]362 /** The resources.
363 * A lazy bird ASSUME there are only two arrays in the resource-fork section in
364 * the XML, namely 'blkx' and 'plst'. These have been assigned fixed indexes. */
[32768]365 DMGUDIFRSRCARRAY aRsrcs[2];
[21401]366 /** The UDIF footer. */
[32768]367 DMGUDIF Ftr;
[32432]368
[32768]369 /** Number of valid extents in the array. */
370 unsigned cExtents;
371 /** Number of entries the array can hold. */
372 unsigned cExtentsMax;
373 /** Pointer to the extent array. */
374 PDMGEXTENT paExtents;
375 /** Index of the last accessed extent. */
376 unsigned idxExtentLast;
377
378 /** Extent which owns the data in the buffer. */
379 PDMGEXTENT pExtentDecomp;
380 /** Buffer holding the decompressed data for a extent. */
381 void *pvDecompExtent;
382 /** Size of the buffer. */
383 size_t cbDecompExtent;
[66486]384 /** The static region list. */
385 VDREGIONLIST RegionList;
[32432]386} DMGIMAGE;
[21401]387/** Pointer to an instance of the DMG Image Interpreter. */
[32432]388typedef DMGIMAGE *PDMGIMAGE;
[21401]389
[32768]390/** @name Resources indexes (into DMG::aRsrcs).
[21401]391 * @{ */
[32768]392#define DMG_RSRC_IDX_BLKX 0
393#define DMG_RSRC_IDX_PLST 1
[21401]394/** @} */
395
[32768]396/** State for the input callout of the inflate reader. */
397typedef struct DMGINFLATESTATE
398{
399 /* Image this operation relates to. */
400 PDMGIMAGE pImage;
401 /* Total size of the data to read. */
402 size_t cbSize;
403 /* Offset in the file to read. */
404 uint64_t uFileOffset;
405 /* Current read position. */
406 ssize_t iOffset;
407} DMGINFLATESTATE;
[21401]408
[57358]409
410/*********************************************************************************************************************************
411* Defined Constants And Macros *
412*********************************************************************************************************************************/
[32768]413/** @def DMG_PRINTF
414 * Wrapper for LogRel.
[21401]415 */
[32768]416#define DMG_PRINTF(a) LogRel(a)
[21401]417
[32768]418/** @def DMG_VALIDATE
[21401]419 * For validating a struct thing and log/print what's wrong.
420 */
[32768]421# define DMG_VALIDATE(expr, logstuff) \
[21401]422 do { \
423 if (!(expr)) \
424 { \
[32768]425 LogRel(("DMG: validation failed: %s\nDMG: ", #expr)); \
[21401]426 LogRel(logstuff); \
427 fRc = false; \
428 } \
429 } while (0)
430
431
[57358]432/*********************************************************************************************************************************
433* Static Variables *
434*********************************************************************************************************************************/
[32536]435
436/** NULL-terminated array of supported file extensions. */
[33524]437static const VDFILEEXTENSION s_aDmgFileExtensions[] =
[32536]438{
[66211]439 {"dmg", VDTYPE_OPTICAL_DISC},
[33524]440 {NULL, VDTYPE_INVALID}
[32536]441};
442
[57358]443
444/*********************************************************************************************************************************
445* Internal Functions *
446*********************************************************************************************************************************/
[62873]447#if 0 /* unused */
[32768]448static void dmgUdifFtrHost2FileEndian(PDMGUDIF pUdif);
[62873]449#endif
[32768]450static void dmgUdifFtrFile2HostEndian(PDMGUDIF pUdif);
[21401]451
[32768]452static void dmgUdifIdHost2FileEndian(PDMGUDIFID pId);
453static void dmgUdifIdFile2HostEndian(PDMGUDIFID pId);
[21401]454
[62873]455#if 0 /* unused */
[32768]456static void dmgUdifCkSumHost2FileEndian(PDMGUDIFCKSUM pCkSum);
[62873]457#endif
[32768]458static void dmgUdifCkSumFile2HostEndian(PDMGUDIFCKSUM pCkSum);
459static bool dmgUdifCkSumIsValid(PCDMGUDIFCKSUM pCkSum, const char *pszPrefix);
[21401]460
[48852]461
462
463/**
464 * vdIfIoIntFileReadSync / RTVfsFileReadAt wrapper.
465 */
466static int dmgWrapFileReadSync(PDMGIMAGE pThis, RTFOFF off, void *pvBuf, size_t cbToRead)
467{
468 int rc;
469 if (pThis->hDmgFileInXar == NIL_RTVFSFILE)
470 rc = vdIfIoIntFileReadSync(pThis->pIfIoXxx, pThis->pStorage, off, pvBuf, cbToRead);
471 else
472 rc = RTVfsFileReadAt(pThis->hDmgFileInXar, off, pvBuf, cbToRead, NULL);
473 return rc;
474}
475
476/**
477 * vdIfIoIntFileReadUser / RTVfsFileReadAt wrapper.
478 */
479static int dmgWrapFileReadUser(PDMGIMAGE pThis, RTFOFF off, PVDIOCTX pIoCtx, size_t cbToRead)
480{
481 int rc;
482 if (pThis->hDmgFileInXar == NIL_RTVFSFILE)
483 rc = vdIfIoIntFileReadUser(pThis->pIfIoXxx, pThis->pStorage, off, pIoCtx, cbToRead);
484 else
485 {
486 /*
[48854]487 * Alloate a temporary buffer on the stack or heap and use
488 * vdIfIoIntIoCtxCopyTo to work the context.
489 *
490 * The I/O context stuff seems too complicated and undocument that I'm
491 * not going to bother trying to implement this efficiently right now.
[48852]492 */
493 void *pvFree = NULL;
494 void *pvBuf;
495 if (cbToRead < _32K)
496 pvBuf = alloca(cbToRead);
497 else
498 pvFree = pvBuf = RTMemTmpAlloc(cbToRead);
[48854]499 if (pvBuf)
[48852]500 {
501 rc = RTVfsFileReadAt(pThis->hDmgFileInXar, off, pvBuf, cbToRead, NULL);
502 if (RT_SUCCESS(rc))
503 vdIfIoIntIoCtxCopyTo(pThis->pIfIoXxx, pIoCtx, pvBuf, cbToRead);
504 if (pvFree)
505 RTMemTmpFree(pvFree);
506 }
507 else
508 rc = VERR_NO_TMP_MEMORY;
509 }
510 return rc;
511}
512
513/**
[80585]514 * vdIfIoIntFileGetSize / RTVfsFileQuerySize wrapper.
[48852]515 */
516static int dmgWrapFileGetSize(PDMGIMAGE pThis, uint64_t *pcbFile)
517{
518 int rc;
519 if (pThis->hDmgFileInXar == NIL_RTVFSFILE)
520 rc = vdIfIoIntFileGetSize(pThis->pIfIoXxx, pThis->pStorage, pcbFile);
521 else
[80585]522 rc = RTVfsFileQuerySize(pThis->hDmgFileInXar, pcbFile);
[48852]523 return rc;
524}
525
526
527
[32768]528static DECLCALLBACK(int) dmgFileInflateHelper(void *pvUser, void *pvBuf, size_t cbBuf, size_t *pcbBuf)
529{
530 DMGINFLATESTATE *pInflateState = (DMGINFLATESTATE *)pvUser;
[32536]531
[32768]532 Assert(cbBuf);
533 if (pInflateState->iOffset < 0)
534 {
535 *(uint8_t *)pvBuf = RTZIPTYPE_ZLIB;
536 if (pcbBuf)
537 *pcbBuf = 1;
538 pInflateState->iOffset = 0;
539 return VINF_SUCCESS;
540 }
541 cbBuf = RT_MIN(cbBuf, pInflateState->cbSize);
[48852]542 int rc = dmgWrapFileReadSync(pInflateState->pImage, pInflateState->uFileOffset, pvBuf, cbBuf);
[32768]543 if (RT_FAILURE(rc))
544 return rc;
545 pInflateState->uFileOffset += cbBuf;
546 pInflateState->iOffset += cbBuf;
547 pInflateState->cbSize -= cbBuf;
548 Assert(pcbBuf);
549 *pcbBuf = cbBuf;
550 return VINF_SUCCESS;
551}
552
[21401]553/**
[32768]554 * Internal: read from a file and inflate the compressed data,
555 * distinguishing between async and normal operation
556 */
557DECLINLINE(int) dmgFileInflateSync(PDMGIMAGE pImage, uint64_t uOffset, size_t cbToRead,
558 void *pvBuf, size_t cbBuf)
559{
560 int rc;
561 PRTZIPDECOMP pZip = NULL;
562 DMGINFLATESTATE InflateState;
563 size_t cbActuallyRead;
564
565 InflateState.pImage = pImage;
566 InflateState.cbSize = cbToRead;
567 InflateState.uFileOffset = uOffset;
568 InflateState.iOffset = -1;
569
570 rc = RTZipDecompCreate(&pZip, &InflateState, dmgFileInflateHelper);
571 if (RT_FAILURE(rc))
572 return rc;
573 rc = RTZipDecompress(pZip, pvBuf, cbBuf, &cbActuallyRead);
574 RTZipDecompDestroy(pZip);
575 if (RT_FAILURE(rc))
576 return rc;
577 if (cbActuallyRead != cbBuf)
578 rc = VERR_VD_VMDK_INVALID_FORMAT;
579 return rc;
580}
581
582/**
[21401]583 * Swaps endian.
584 * @param pUdif The structure.
585 */
[32768]586static void dmgSwapEndianUdif(PDMGUDIF pUdif)
[21401]587{
588#ifndef RT_BIG_ENDIAN
589 pUdif->u32Magic = RT_BSWAP_U32(pUdif->u32Magic);
590 pUdif->u32Version = RT_BSWAP_U32(pUdif->u32Version);
591 pUdif->cbFooter = RT_BSWAP_U32(pUdif->cbFooter);
592 pUdif->fFlags = RT_BSWAP_U32(pUdif->fFlags);
593 pUdif->offRunData = RT_BSWAP_U64(pUdif->offRunData);
594 pUdif->offData = RT_BSWAP_U64(pUdif->offData);
595 pUdif->cbData = RT_BSWAP_U64(pUdif->cbData);
596 pUdif->offRsrc = RT_BSWAP_U64(pUdif->offRsrc);
597 pUdif->cbRsrc = RT_BSWAP_U64(pUdif->cbRsrc);
598 pUdif->iSegment = RT_BSWAP_U32(pUdif->iSegment);
599 pUdif->cSegments = RT_BSWAP_U32(pUdif->cSegments);
600 pUdif->offXml = RT_BSWAP_U64(pUdif->offXml);
601 pUdif->cbXml = RT_BSWAP_U64(pUdif->cbXml);
602 pUdif->u32Type = RT_BSWAP_U32(pUdif->u32Type);
603 pUdif->cSectors = RT_BSWAP_U64(pUdif->cSectors);
604#endif
605}
606
607
[62873]608#if 0 /* unused */
[21401]609/**
610 * Swaps endian from host cpu to file.
611 * @param pUdif The structure.
612 */
[32768]613static void dmgUdifFtrHost2FileEndian(PDMGUDIF pUdif)
[21401]614{
[32768]615 dmgSwapEndianUdif(pUdif);
616 dmgUdifIdHost2FileEndian(&pUdif->SegmentId);
617 dmgUdifCkSumHost2FileEndian(&pUdif->DataCkSum);
618 dmgUdifCkSumHost2FileEndian(&pUdif->MasterCkSum);
[21401]619}
[62873]620#endif
[21401]621
622
623/**
624 * Swaps endian from file to host cpu.
625 * @param pUdif The structure.
626 */
[32768]627static void dmgUdifFtrFile2HostEndian(PDMGUDIF pUdif)
[21401]628{
[32768]629 dmgSwapEndianUdif(pUdif);
630 dmgUdifIdFile2HostEndian(&pUdif->SegmentId);
631 dmgUdifCkSumFile2HostEndian(&pUdif->DataCkSum);
632 dmgUdifCkSumFile2HostEndian(&pUdif->MasterCkSum);
[21401]633}
634
[32432]635/**
636 * Swaps endian from file to host cpu.
637 * @param pBlkx The blkx structure.
638 */
[32768]639static void dmgBlkxFile2HostEndian(PDMGBLKX pBlkx)
[32432]640{
641 pBlkx->u32Magic = RT_BE2H_U32(pBlkx->u32Magic);
642 pBlkx->u32Version = RT_BE2H_U32(pBlkx->u32Version);
643 pBlkx->cSectornumberFirst = RT_BE2H_U64(pBlkx->cSectornumberFirst);
644 pBlkx->cSectors = RT_BE2H_U64(pBlkx->cSectors);
645 pBlkx->offDataStart = RT_BE2H_U64(pBlkx->offDataStart);
646 pBlkx->cSectorsDecompress = RT_BE2H_U32(pBlkx->cSectorsDecompress);
647 pBlkx->u32BlocksDescriptor = RT_BE2H_U32(pBlkx->u32BlocksDescriptor);
648 pBlkx->cBlocksRunCount = RT_BE2H_U32(pBlkx->cBlocksRunCount);
[32768]649 dmgUdifCkSumFile2HostEndian(&pBlkx->BlkxCkSum);
[32432]650}
[21401]651
652/**
[32432]653 * Swaps endian from file to host cpu.
654 * @param pBlkxDesc The blkx descriptor structure.
655 */
[32768]656static void dmgBlkxDescFile2HostEndian(PDMGBLKXDESC pBlkxDesc)
[32432]657{
658 pBlkxDesc->u32Type = RT_BE2H_U32(pBlkxDesc->u32Type);
659 pBlkxDesc->u32Reserved = RT_BE2H_U32(pBlkxDesc->u32Reserved);
660 pBlkxDesc->u64SectorStart = RT_BE2H_U64(pBlkxDesc->u64SectorStart);
661 pBlkxDesc->u64SectorCount = RT_BE2H_U64(pBlkxDesc->u64SectorCount);
662 pBlkxDesc->offData = RT_BE2H_U64(pBlkxDesc->offData);
663 pBlkxDesc->cbData = RT_BE2H_U64(pBlkxDesc->cbData);
664}
665
666/**
[21401]667 * Validates an UDIF footer structure.
668 *
669 * @returns true if valid, false and LogRel()s on failure.
670 * @param pFtr The UDIF footer to validate.
671 * @param offFtr The offset of the structure.
672 */
[32768]673static bool dmgUdifFtrIsValid(PCDMGUDIF pFtr, uint64_t offFtr)
[21401]674{
675 bool fRc = true;
676
[32768]677 DMG_VALIDATE(!(pFtr->fFlags & ~DMGUDIF_FLAGS_KNOWN_MASK), ("fFlags=%#RX32 fKnown=%RX32\n", pFtr->fFlags, DMGUDIF_FLAGS_KNOWN_MASK));
678 DMG_VALIDATE(pFtr->offRunData < offFtr, ("offRunData=%#RX64\n", pFtr->offRunData));
679 DMG_VALIDATE(pFtr->cbData <= offFtr && pFtr->offData + pFtr->cbData <= offFtr, ("cbData=%#RX64 offData=%#RX64 offFtr=%#RX64\n", pFtr->cbData, pFtr->offData, offFtr));
680 DMG_VALIDATE(pFtr->offData < offFtr, ("offData=%#RX64\n", pFtr->offData));
681 DMG_VALIDATE(pFtr->cbRsrc <= offFtr && pFtr->offRsrc + pFtr->cbRsrc <= offFtr, ("cbRsrc=%#RX64 offRsrc=%#RX64 offFtr=%#RX64\n", pFtr->cbRsrc, pFtr->offRsrc, offFtr));
682 DMG_VALIDATE(pFtr->offRsrc < offFtr, ("offRsrc=%#RX64\n", pFtr->offRsrc));
683 DMG_VALIDATE(pFtr->cSegments <= 1, ("cSegments=%RU32\n", pFtr->cSegments));
684 DMG_VALIDATE(pFtr->iSegment == 0 || pFtr->iSegment == 1, ("iSegment=%RU32 cSegments=%RU32\n", pFtr->iSegment, pFtr->cSegments));
685 DMG_VALIDATE(pFtr->cbXml <= offFtr && pFtr->offXml + pFtr->cbXml <= offFtr, ("cbXml=%#RX64 offXml=%#RX64 offFtr=%#RX64\n", pFtr->cbXml, pFtr->offXml, offFtr));
686 DMG_VALIDATE(pFtr->offXml < offFtr, ("offXml=%#RX64\n", pFtr->offXml));
687 DMG_VALIDATE(pFtr->cbXml > 128, ("cbXml=%#RX64\n", pFtr->cbXml));
688 DMG_VALIDATE(pFtr->cbXml < 10 * _1M, ("cbXml=%#RX64\n", pFtr->cbXml));
689 DMG_VALIDATE(pFtr->u32Type == DMGUDIF_TYPE_DEVICE || pFtr->u32Type == DMGUDIF_TYPE_PARTITION, ("u32Type=%RU32\n", pFtr->u32Type));
690 DMG_VALIDATE(pFtr->cSectors != 0, ("cSectors=%#RX64\n", pFtr->cSectors));
691 fRc &= dmgUdifCkSumIsValid(&pFtr->DataCkSum, "DataCkSum");
692 fRc &= dmgUdifCkSumIsValid(&pFtr->MasterCkSum, "MasterCkSum");
[21401]693
694 return fRc;
695}
696
697
[32768]698static bool dmgBlkxIsValid(PCDMGBLKX pBlkx)
[32432]699{
700 bool fRc = true;
701
[32768]702 fRc &= dmgUdifCkSumIsValid(&pBlkx->BlkxCkSum, "BlkxCkSum");
703 DMG_VALIDATE(pBlkx->u32Magic == DMGBLKX_MAGIC, ("u32Magic=%#RX32 u32MagicExpected=%#RX32\n", pBlkx->u32Magic, DMGBLKX_MAGIC));
704 DMG_VALIDATE(pBlkx->u32Version == DMGBLKX_VERSION, ("u32Version=%#RX32 u32VersionExpected=%#RX32\n", pBlkx->u32Magic, DMGBLKX_VERSION));
[32432]705
706 return fRc;
707}
708
[21401]709/**
710 * Swaps endian from host cpu to file.
711 * @param pId The structure.
712 */
[32768]713static void dmgUdifIdHost2FileEndian(PDMGUDIFID pId)
[21401]714{
715 NOREF(pId);
716}
717
718
719/**
720 * Swaps endian from file to host cpu.
721 * @param pId The structure.
722 */
[32768]723static void dmgUdifIdFile2HostEndian(PDMGUDIFID pId)
[21401]724{
[32768]725 dmgUdifIdHost2FileEndian(pId);
[21401]726}
727
728
729/**
730 * Swaps endian.
731 * @param pCkSum The structure.
[64272]732 * @param u32Kind Kind of the checksum (CRC32, none)
733 * @param cBits Size of the checksum in bits.
[21401]734 */
[32768]735static void dmgSwapEndianUdifCkSum(PDMGUDIFCKSUM pCkSum, uint32_t u32Kind, uint32_t cBits)
[21401]736{
737#ifdef RT_BIG_ENDIAN
738 NOREF(pCkSum);
739 NOREF(u32Kind);
740 NOREF(cBits);
741#else
742 switch (u32Kind)
743 {
[32768]744 case DMGUDIFCKSUM_NONE:
[21401]745 /* nothing to do here */
746 break;
747
[32768]748 case DMGUDIFCKSUM_CRC32:
[21401]749 Assert(cBits == 32);
750 pCkSum->u32Kind = RT_BSWAP_U32(pCkSum->u32Kind);
751 pCkSum->cBits = RT_BSWAP_U32(pCkSum->cBits);
752 pCkSum->uSum.au32[0] = RT_BSWAP_U32(pCkSum->uSum.au32[0]);
753 break;
754
755 default:
[33816]756 AssertMsgFailed(("%x\n", u32Kind));
[21401]757 break;
758 }
759 NOREF(cBits);
760#endif
761}
762
763
[62873]764#if 0 /* unused */
[21401]765/**
766 * Swaps endian from host cpu to file.
767 * @param pCkSum The structure.
768 */
[32768]769static void dmgUdifCkSumHost2FileEndian(PDMGUDIFCKSUM pCkSum)
[21401]770{
[32768]771 dmgSwapEndianUdifCkSum(pCkSum, pCkSum->u32Kind, pCkSum->cBits);
[21401]772}
[62873]773#endif
[21401]774
775
776/**
777 * Swaps endian from file to host cpu.
778 * @param pCkSum The structure.
779 */
[32768]780static void dmgUdifCkSumFile2HostEndian(PDMGUDIFCKSUM pCkSum)
[21401]781{
[32768]782 dmgSwapEndianUdifCkSum(pCkSum, RT_BE2H_U32(pCkSum->u32Kind), RT_BE2H_U32(pCkSum->cBits));
[21401]783}
784
785
786/**
787 * Validates an UDIF checksum structure.
788 *
789 * @returns true if valid, false and LogRel()s on failure.
790 * @param pCkSum The checksum structure.
791 * @param pszPrefix The message prefix.
792 * @remarks This does not check the checksummed data.
793 */
[32768]794static bool dmgUdifCkSumIsValid(PCDMGUDIFCKSUM pCkSum, const char *pszPrefix)
[21401]795{
796 bool fRc = true;
797
798 switch (pCkSum->u32Kind)
799 {
[32768]800 case DMGUDIFCKSUM_NONE:
801 DMG_VALIDATE(pCkSum->cBits == 0, ("%s/NONE: cBits=%d\n", pszPrefix, pCkSum->cBits));
[21401]802 break;
803
[32768]804 case DMGUDIFCKSUM_CRC32:
805 DMG_VALIDATE(pCkSum->cBits == 32, ("%s/NONE: cBits=%d\n", pszPrefix, pCkSum->cBits));
[21401]806 break;
807
808 default:
[32768]809 DMG_VALIDATE(0, ("%s: u32Kind=%#RX32\n", pszPrefix, pCkSum->u32Kind));
[21401]810 break;
811 }
812 return fRc;
813}
814
815
[32536]816/**
817 * Internal. Flush image data to disk.
818 */
819static int dmgFlushImage(PDMGIMAGE pThis)
[21401]820{
[32536]821 int rc = VINF_SUCCESS;
[21401]822
[48871]823 if ( pThis
824 && (pThis->pStorage || pThis->hDmgFileInXar != NIL_RTVFSFILE)
[32536]825 && !(pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY))
[21401]826 {
[48871]827 /** @todo handle writable files, update checksums etc. */
[21401]828 }
829
[32536]830 return rc;
831}
832
833
834/**
835 * Internal. Free all allocated space for representing an image except pThis,
836 * and optionally delete the image from disk.
837 */
838static int dmgFreeImage(PDMGIMAGE pThis, bool fDelete)
839{
[21401]840 int rc = VINF_SUCCESS;
841
[32536]842 /* Freeing a never allocated image (e.g. because the open failed) is
843 * not signalled as an error. After all nothing bad happens. */
844 if (pThis)
845 {
[48871]846 RTVfsFileRelease(pThis->hDmgFileInXar);
847 pThis->hDmgFileInXar = NIL_RTVFSFILE;
848
849 RTVfsFsStrmRelease(pThis->hXarFss);
850 pThis->hXarFss = NIL_RTVFSFSSTREAM;
851
[32536]852 if (pThis->pStorage)
[21401]853 {
[32536]854 /* No point updating the file that is deleted anyway. */
855 if (!fDelete)
856 dmgFlushImage(pThis);
857
[48852]858 rc = vdIfIoIntFileClose(pThis->pIfIoXxx, pThis->pStorage);
[32536]859 pThis->pStorage = NULL;
860 }
861
862 for (unsigned iRsrc = 0; iRsrc < RT_ELEMENTS(pThis->aRsrcs); iRsrc++)
863 for (unsigned i = 0; i < pThis->aRsrcs[iRsrc].cEntries; i++)
[21401]864 {
[32536]865 if (pThis->aRsrcs[iRsrc].aEntries[i].pbData)
866 {
867 RTMemFree(pThis->aRsrcs[iRsrc].aEntries[i].pbData);
868 pThis->aRsrcs[iRsrc].aEntries[i].pbData = NULL;
869 }
870 if (pThis->aRsrcs[iRsrc].aEntries[i].pszName)
871 {
872 RTMemFree(pThis->aRsrcs[iRsrc].aEntries[i].pszName);
873 pThis->aRsrcs[iRsrc].aEntries[i].pszName = NULL;
874 }
875 if (pThis->aRsrcs[iRsrc].aEntries[i].pszCFName)
876 {
877 RTMemFree(pThis->aRsrcs[iRsrc].aEntries[i].pszCFName);
878 pThis->aRsrcs[iRsrc].aEntries[i].pszCFName = NULL;
879 }
[21401]880 }
881
[32536]882 if (fDelete && pThis->pszFilename)
[48852]883 vdIfIoIntFileDelete(pThis->pIfIoXxx, pThis->pszFilename);
[32768]884
885 if (pThis->pvDecompExtent)
886 {
887 RTMemFree(pThis->pvDecompExtent);
888 pThis->pvDecompExtent = NULL;
889 pThis->cbDecompExtent = 0;
890 }
[61971]891
892 if (pThis->paExtents)
893 {
894 RTMemFree(pThis->paExtents);
895 pThis->paExtents = NULL;
896 }
[32536]897 }
898
899 LogFlowFunc(("returns %Rrc\n", rc));
[21401]900 return rc;
901}
902
903
904#define STARTS_WITH(pszString, szStart) \
905 ( strncmp(pszString, szStart, sizeof(szStart) - 1) == 0 )
906
907#define STARTS_WITH_WORD(pszString, szWord) \
908 ( STARTS_WITH(pszString, szWord) \
909 && !RT_C_IS_ALNUM((pszString)[sizeof(szWord) - 1]) )
910
911#define SKIP_AHEAD(psz, szWord) \
912 do { \
913 (psz) = RTStrStripL((psz) + sizeof(szWord) - 1); \
914 } while (0)
915
916#define REQUIRE_WORD(psz, szWord) \
917 do { \
918 if (!STARTS_WITH_WORD(psz, szWord)) \
919 return psz; \
920 (psz) = RTStrStripL((psz) + sizeof(szWord) - 1); \
921 } while (0)
922
923#define REQUIRE_TAG(psz, szTag) \
924 do { \
925 if (!STARTS_WITH(psz, "<" szTag ">")) \
926 return psz; \
927 (psz) = RTStrStripL((psz) + sizeof("<" szTag ">") - 1); \
928 } while (0)
929
930#define REQUIRE_TAG_NO_STRIP(psz, szTag) \
931 do { \
932 if (!STARTS_WITH(psz, "<" szTag ">")) \
933 return psz; \
934 (psz) += sizeof("<" szTag ">") - 1; \
935 } while (0)
936
937#define REQUIRE_END_TAG(psz, szTag) \
938 do { \
939 if (!STARTS_WITH(psz, "</" szTag ">")) \
940 return psz; \
941 (psz) = RTStrStripL((psz) + sizeof("</" szTag ">") - 1); \
942 } while (0)
943
944
945/**
946 * Finds the next tag end.
947 *
948 * @returns Pointer to a '>' or '\0'.
949 * @param pszCur The current position.
950 */
[32768]951static const char *dmgXmlFindTagEnd(const char *pszCur)
[21401]952{
953 /* Might want to take quoted '>' into account? */
954 char ch;
955 while ((ch = *pszCur) != '\0' && ch != '>')
956 pszCur++;
957 return pszCur;
958}
959
960
961/**
962 * Finds the end tag.
963 *
[64272]964 * Does not deal with @verbatim<tag attr="1"/>@endverbatim style tags.
[21401]965 *
966 * @returns Pointer to the first char in the end tag. NULL if another tag
967 * was encountered first or if we hit the end of the file.
968 * @param ppszCur The current position (IN/OUT).
969 * @param pszTag The tag name.
970 */
[32768]971static const char *dmgXmlFindEndTag(const char **ppszCur, const char *pszTag)
[21401]972{
973 const char *psz = *ppszCur;
974 char ch;
975 while ((ch = *psz))
976 {
977 if (ch == '<')
978 {
979 size_t const cchTag = strlen(pszTag);
980 if ( psz[1] == '/'
981 && !memcmp(&psz[2], pszTag, cchTag)
982 && psz[2 + cchTag] == '>')
983 {
984 *ppszCur = psz + 2 + cchTag + 1;
985 return psz;
986 }
987 break;
988 }
989 psz++;
990 }
991 return NULL;
992}
993
994
995/**
996 * Reads a signed 32-bit value.
997 *
998 * @returns NULL on success, pointer to the offending text on failure.
999 * @param ppszCur The text position (IN/OUT).
1000 * @param pi32 Where to store the value.
1001 */
[32768]1002static const char *dmgXmlParseS32(const char **ppszCur, int32_t *pi32)
[21401]1003{
1004 const char *psz = *ppszCur;
1005
1006 /*
1007 * <string>-1</string>
1008 */
1009 REQUIRE_TAG_NO_STRIP(psz, "string");
1010
1011 char *pszNext;
1012 int rc = RTStrToInt32Ex(psz, &pszNext, 0, pi32);
1013 if (rc != VWRN_TRAILING_CHARS)
1014 return *ppszCur;
1015 psz = pszNext;
1016
1017 REQUIRE_END_TAG(psz, "string");
1018 *ppszCur = psz;
1019 return NULL;
1020}
1021
1022
1023/**
1024 * Reads an unsigned 32-bit value.
1025 *
1026 * @returns NULL on success, pointer to the offending text on failure.
1027 * @param ppszCur The text position (IN/OUT).
1028 * @param pu32 Where to store the value.
1029 */
[32768]1030static const char *dmgXmlParseU32(const char **ppszCur, uint32_t *pu32)
[21401]1031{
1032 const char *psz = *ppszCur;
1033
1034 /*
1035 * <string>0x00ff</string>
1036 */
1037 REQUIRE_TAG_NO_STRIP(psz, "string");
1038
1039 char *pszNext;
1040 int rc = RTStrToUInt32Ex(psz, &pszNext, 0, pu32);
1041 if (rc != VWRN_TRAILING_CHARS)
1042 return *ppszCur;
1043 psz = pszNext;
1044
1045 REQUIRE_END_TAG(psz, "string");
1046 *ppszCur = psz;
1047 return NULL;
1048}
1049
1050
1051/**
1052 * Reads a string value.
1053 *
1054 * @returns NULL on success, pointer to the offending text on failure.
1055 * @param ppszCur The text position (IN/OUT).
1056 * @param ppszString Where to store the pointer to the string. The caller
1057 * must free this using RTMemFree.
1058 */
[32768]1059static const char *dmgXmlParseString(const char **ppszCur, char **ppszString)
[21401]1060{
1061 const char *psz = *ppszCur;
1062
1063 /*
1064 * <string>Driver Descriptor Map (DDM : 0)</string>
1065 */
1066 REQUIRE_TAG_NO_STRIP(psz, "string");
1067
1068 const char *pszStart = psz;
[32768]1069 const char *pszEnd = dmgXmlFindEndTag(&psz, "string");
[21401]1070 if (!pszEnd)
1071 return *ppszCur;
1072 psz = RTStrStripL(psz);
1073
1074 *ppszString = (char *)RTMemDupEx(pszStart, pszEnd - pszStart, 1);
1075 if (!*ppszString)
1076 return *ppszCur;
1077
1078 *ppszCur = psz;
1079 return NULL;
1080}
1081
1082
1083/**
1084 * Parses the BASE-64 coded data tags.
1085 *
1086 * @returns NULL on success, pointer to the offending text on failure.
1087 * @param ppszCur The text position (IN/OUT).
1088 * @param ppbData Where to store the pointer to the data we've read. The
1089 * caller must free this using RTMemFree.
1090 * @param pcbData The number of bytes we're returning.
1091 */
[32768]1092static const char *dmgXmlParseData(const char **ppszCur, uint8_t **ppbData, size_t *pcbData)
[21401]1093{
1094 const char *psz = *ppszCur;
1095
1096 /*
1097 * <data> AAAAA... </data>
1098 */
1099 REQUIRE_TAG(psz, "data");
1100
1101 const char *pszStart = psz;
1102 ssize_t cbData = RTBase64DecodedSize(pszStart, (char **)&psz);
1103 if (cbData == -1)
1104 return *ppszCur;
1105
1106 REQUIRE_END_TAG(psz, "data");
1107
1108 *ppbData = (uint8_t *)RTMemAlloc(cbData);
1109 if (!*ppbData)
1110 return *ppszCur;
1111 char *pszIgnored;
1112 int rc = RTBase64Decode(pszStart, *ppbData, cbData, pcbData, &pszIgnored);
1113 if (RT_FAILURE(rc))
1114 {
1115 RTMemFree(*ppbData);
1116 *ppbData = NULL;
1117 return *ppszCur;
1118 }
1119
1120 *ppszCur = psz;
1121 return NULL;
1122}
1123
1124
1125/**
1126 * Parses the XML resource-fork in a rather presumptive manner.
1127 *
[32768]1128 * This function is supposed to construct the DMG::aRsrcs instance data
[21401]1129 * parts.
1130 *
1131 * @returns NULL on success, pointer to the problematic text on failure.
1132 * @param pThis The DMG instance data.
1133 * @param pszXml The XML text to parse, UTF-8.
1134 */
[32768]1135static const char *dmgOpenXmlToRsrc(PDMGIMAGE pThis, char const *pszXml)
[21401]1136{
1137 const char *psz = pszXml;
1138
1139 /*
1140 * Verify the ?xml, !DOCTYPE and plist tags.
1141 */
1142 SKIP_AHEAD(psz, "");
1143
1144 /* <?xml version="1.0" encoding="UTF-8"?> */
1145 REQUIRE_WORD(psz, "<?xml");
1146 while (*psz != '?')
1147 {
1148 if (!*psz)
1149 return psz;
1150 if (STARTS_WITH_WORD(psz, "version="))
1151 {
1152 SKIP_AHEAD(psz, "version=");
1153 REQUIRE_WORD(psz, "\"1.0\"");
1154 }
1155 else if (STARTS_WITH_WORD(psz, "encoding="))
1156 {
1157 SKIP_AHEAD(psz, "encoding=");
1158 REQUIRE_WORD(psz, "\"UTF-8\"");
1159 }
1160 else
1161 return psz;
1162 }
1163 SKIP_AHEAD(psz, "?>");
1164
1165 /* <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd"> */
1166 REQUIRE_WORD(psz, "<!DOCTYPE");
1167 REQUIRE_WORD(psz, "plist");
1168 REQUIRE_WORD(psz, "PUBLIC");
[32768]1169 psz = dmgXmlFindTagEnd(psz);
[21401]1170 REQUIRE_WORD(psz, ">");
1171
1172 /* <plist version="1.0"> */
1173 REQUIRE_WORD(psz, "<plist");
1174 REQUIRE_WORD(psz, "version=");
1175 REQUIRE_WORD(psz, "\"1.0\"");
1176 REQUIRE_WORD(psz, ">");
1177
1178 /*
[33540]1179 * Descend down to the 'resource-fork' dictionary.
[21401]1180 * ASSUME it's the only top level dictionary.
1181 */
1182 /* <dict> <key>resource-fork</key> */
1183 REQUIRE_TAG(psz, "dict");
1184 REQUIRE_WORD(psz, "<key>resource-fork</key>");
1185
1186 /*
1187 * Parse the keys in the resource-fork dictionary.
1188 * ASSUME that there are just two, 'blkx' and 'plst'.
1189 */
1190 REQUIRE_TAG(psz, "dict");
1191 while (!STARTS_WITH_WORD(psz, "</dict>"))
1192 {
1193 /*
1194 * Parse the key and Create the resource-fork entry.
1195 */
1196 unsigned iRsrc;
1197 if (STARTS_WITH_WORD(psz, "<key>blkx</key>"))
1198 {
1199 REQUIRE_WORD(psz, "<key>blkx</key>");
[32768]1200 iRsrc = DMG_RSRC_IDX_BLKX;
[21401]1201 strcpy(&pThis->aRsrcs[iRsrc].szName[0], "blkx");
1202 }
1203 else if (STARTS_WITH_WORD(psz, "<key>plst</key>"))
1204 {
1205 REQUIRE_WORD(psz, "<key>plst</key>");
[32768]1206 iRsrc = DMG_RSRC_IDX_PLST;
[21401]1207 strcpy(&pThis->aRsrcs[iRsrc].szName[0], "plst");
1208 }
1209 else
[33547]1210 {
1211 SKIP_AHEAD(psz, "</array>");
1212 continue;
1213 }
[21401]1214
[33547]1215
[21401]1216 /*
[33540]1217 * Descend into the array and add the elements to the resource entry.
[21401]1218 */
1219 /* <array> */
1220 REQUIRE_TAG(psz, "array");
1221 while (!STARTS_WITH_WORD(psz, "</array>"))
1222 {
1223 REQUIRE_TAG(psz, "dict");
1224 uint32_t i = pThis->aRsrcs[iRsrc].cEntries;
1225 if (i == RT_ELEMENTS(pThis->aRsrcs[iRsrc].aEntries))
1226 return psz;
1227
1228 while (!STARTS_WITH_WORD(psz, "</dict>"))
1229 {
1230
1231 /* switch on the key. */
1232 const char *pszErr;
1233 if (STARTS_WITH_WORD(psz, "<key>Attributes</key>"))
1234 {
1235 REQUIRE_WORD(psz, "<key>Attributes</key>");
[32768]1236 pszErr = dmgXmlParseU32(&psz, &pThis->aRsrcs[iRsrc].aEntries[i].fAttributes);
[21401]1237 }
1238 else if (STARTS_WITH_WORD(psz, "<key>ID</key>"))
1239 {
1240 REQUIRE_WORD(psz, "<key>ID</key>");
[32768]1241 pszErr = dmgXmlParseS32(&psz, &pThis->aRsrcs[iRsrc].aEntries[i].iId);
[21401]1242 }
1243 else if (STARTS_WITH_WORD(psz, "<key>Name</key>"))
1244 {
1245 REQUIRE_WORD(psz, "<key>Name</key>");
[32768]1246 pszErr = dmgXmlParseString(&psz, &pThis->aRsrcs[iRsrc].aEntries[i].pszName);
[21401]1247 }
1248 else if (STARTS_WITH_WORD(psz, "<key>CFName</key>"))
1249 {
1250 REQUIRE_WORD(psz, "<key>CFName</key>");
[32768]1251 pszErr = dmgXmlParseString(&psz, &pThis->aRsrcs[iRsrc].aEntries[i].pszCFName);
[21401]1252 }
1253 else if (STARTS_WITH_WORD(psz, "<key>Data</key>"))
1254 {
1255 REQUIRE_WORD(psz, "<key>Data</key>");
[32768]1256 pszErr = dmgXmlParseData(&psz, &pThis->aRsrcs[iRsrc].aEntries[i].pbData, &pThis->aRsrcs[iRsrc].aEntries[i].cbData);
[21401]1257 }
1258 else
1259 pszErr = psz;
1260 if (pszErr)
1261 return pszErr;
1262 } /* while not </dict> */
1263 REQUIRE_END_TAG(psz, "dict");
1264
[32432]1265 pThis->aRsrcs[iRsrc].cEntries++;
[21401]1266 } /* while not </array> */
1267 REQUIRE_END_TAG(psz, "array");
1268
1269 } /* while not </dict> */
1270 REQUIRE_END_TAG(psz, "dict");
1271
1272 /*
1273 * ASSUMING there is only the 'resource-fork', we'll now see the end of
1274 * the outer dict, plist and text.
1275 */
1276 /* </dict> </plist> */
1277 REQUIRE_END_TAG(psz, "dict");
1278 REQUIRE_END_TAG(psz, "plist");
1279
1280 /* the end */
1281 if (*psz)
1282 return psz;
1283
1284 return NULL;
1285}
1286
1287#undef REQUIRE_END_TAG
1288#undef REQUIRE_TAG_NO_STRIP
1289#undef REQUIRE_TAG
1290#undef REQUIRE_WORD
1291#undef SKIP_AHEAD
1292#undef STARTS_WITH_WORD
1293#undef STARTS_WITH
1294
[32432]1295/**
1296 * Returns the data attached to a resource.
1297 *
1298 * @returns VBox status code.
1299 * @param pThis The DMG instance data.
1300 * @param pcszRsrcName Name of the resource to get.
[64272]1301 * @param ppcRsrc Where to store the pointer to the resource data on success.
[32432]1302 */
1303static int dmgGetRsrcData(PDMGIMAGE pThis, const char *pcszRsrcName,
[32768]1304 PCDMGUDIFRSRCARRAY *ppcRsrc)
[32432]1305{
1306 int rc = VERR_NOT_FOUND;
[21401]1307
[32432]1308 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aRsrcs); i++)
1309 {
1310 if (!strcmp(pThis->aRsrcs[i].szName, pcszRsrcName))
1311 {
1312 *ppcRsrc = &pThis->aRsrcs[i];
1313 rc = VINF_SUCCESS;
1314 break;
1315 }
1316 }
1317
1318 return rc;
1319}
1320
[21401]1321/**
[32432]1322 * Creates a new extent from the given blkx descriptor.
1323 *
1324 * @returns VBox status code.
[32768]1325 * @param pThis DMG instance data.
1326 * @param uSectorPart First sector the partition owning the blkx descriptor has.
1327 * @param pBlkxDesc The blkx descriptor.
[32432]1328 */
[32768]1329static int dmgExtentCreateFromBlkxDesc(PDMGIMAGE pThis, uint64_t uSectorPart, PDMGBLKXDESC pBlkxDesc)
[32432]1330{
1331 int rc = VINF_SUCCESS;
1332 DMGEXTENTTYPE enmExtentTypeNew;
1333 PDMGEXTENT pExtentNew = NULL;
1334
[32768]1335 if (pBlkxDesc->u32Type == DMGBLKXDESC_TYPE_RAW)
[32432]1336 enmExtentTypeNew = DMGEXTENTTYPE_RAW;
[32768]1337 else if (pBlkxDesc->u32Type == DMGBLKXDESC_TYPE_IGNORE)
[32432]1338 enmExtentTypeNew = DMGEXTENTTYPE_ZERO;
[32768]1339 else if (pBlkxDesc->u32Type == DMGBLKXDESC_TYPE_ZLIB)
1340 enmExtentTypeNew = DMGEXTENTTYPE_COMP_ZLIB;
[32432]1341 else
[32768]1342 {
[32432]1343 AssertMsgFailed(("This method supports only raw or zero extents!\n"));
[32768]1344 return VERR_NOT_SUPPORTED;
1345 }
[32432]1346
[63567]1347 /** @todo Merge raw extents if possible to save memory. */
[32432]1348#if 0
1349 pExtentNew = pThis->pExtentLast;
1350 if ( pExtentNew
1351 && pExtentNew->enmType == enmExtentTypeNew
[32768]1352 && enmExtentTypeNew == DMGEXTENTTYPE_RAW
1353 && pExtentNew->uSectorExtent + pExtentNew->cSectorsExtent == offDevice + pBlkxDesc->u64SectorStart * DMG_SECTOR_SIZE;
[32432]1354 && pExtentNew->offFileStart + pExtentNew->cbExtent == pBlkxDesc->offData)
1355 {
1356 /* Increase the last extent. */
1357 pExtentNew->cbExtent += pBlkxDesc->cbData;
1358 }
1359 else
1360#endif
1361 {
[32768]1362 if (pThis->cExtentsMax == pThis->cExtents)
[32432]1363 {
[32768]1364 pThis->cExtentsMax += 64;
[32432]1365
[32768]1366 /* Increase the array. */
1367 PDMGEXTENT paExtentsNew = (PDMGEXTENT)RTMemRealloc(pThis->paExtents, sizeof(DMGEXTENT) * pThis->cExtentsMax);
1368 if (!paExtentsNew)
[32432]1369 {
[32768]1370 rc = VERR_NO_MEMORY;
1371 pThis->cExtentsMax -= 64;
[32432]1372 }
1373 else
[32768]1374 pThis->paExtents = paExtentsNew;
[32432]1375 }
[32768]1376
1377 if (RT_SUCCESS(rc))
1378 {
1379 pExtentNew = &pThis->paExtents[pThis->cExtents++];
1380
1381 pExtentNew->enmType = enmExtentTypeNew;
1382 pExtentNew->uSectorExtent = uSectorPart + pBlkxDesc->u64SectorStart;
1383 pExtentNew->cSectorsExtent = pBlkxDesc->u64SectorCount;
1384 pExtentNew->offFileStart = pBlkxDesc->offData;
1385 pExtentNew->cbFile = pBlkxDesc->cbData;
1386 }
[32432]1387 }
1388
1389 return rc;
1390}
1391
1392/**
[32768]1393 * Find the extent for the given sector number.
[32432]1394 */
[32768]1395static PDMGEXTENT dmgExtentGetFromOffset(PDMGIMAGE pThis, uint64_t uSector)
[32432]1396{
[32768]1397 /*
1398 * We assume that the array is ordered from lower to higher sector
1399 * numbers.
1400 * This makes it possible to bisect the array to find the extent
1401 * faster than using a linked list.
1402 */
1403 PDMGEXTENT pExtent = NULL;
1404 unsigned idxCur = pThis->idxExtentLast;
1405 unsigned idxMax = pThis->cExtents;
1406 unsigned idxMin = 0;
[32432]1407
[32768]1408 while (idxMin < idxMax)
1409 {
1410 PDMGEXTENT pExtentCur = &pThis->paExtents[idxCur];
[32432]1411
[32768]1412 /* Determine the search direction. */
1413 if (uSector < pExtentCur->uSectorExtent)
1414 {
1415 /* Search left from the current extent. */
1416 idxMax = idxCur;
1417 }
1418 else if (uSector >= pExtentCur->uSectorExtent + pExtentCur->cSectorsExtent)
1419 {
1420 /* Search right from the current extent. */
1421 idxMin = idxCur;
1422 }
1423 else
1424 {
1425 /* The sector lies in the extent, stop searching. */
1426 pExtent = pExtentCur;
1427 break;
1428 }
1429
1430 idxCur = idxMin + (idxMax - idxMin) / 2;
1431 }
1432
1433 if (pExtent)
1434 pThis->idxExtentLast = idxCur;
1435
[32432]1436 return pExtent;
1437}
1438
1439/**
1440 * Goes through the BLKX structure and creates the necessary extents.
1441 */
[32768]1442static int dmgBlkxParse(PDMGIMAGE pThis, PDMGBLKX pBlkx)
[32432]1443{
1444 int rc = VINF_SUCCESS;
[32768]1445 PDMGBLKXDESC pBlkxDesc = (PDMGBLKXDESC)(pBlkx + 1);
[32432]1446
1447 for (unsigned i = 0; i < pBlkx->cBlocksRunCount; i++)
1448 {
[32768]1449 dmgBlkxDescFile2HostEndian(pBlkxDesc);
[32432]1450
1451 switch (pBlkxDesc->u32Type)
1452 {
[32768]1453 case DMGBLKXDESC_TYPE_RAW:
1454 case DMGBLKXDESC_TYPE_IGNORE:
1455 case DMGBLKXDESC_TYPE_ZLIB:
[32432]1456 {
[32768]1457 rc = dmgExtentCreateFromBlkxDesc(pThis, pBlkx->cSectornumberFirst, pBlkxDesc);
[32432]1458 break;
1459 }
[32768]1460 case DMGBLKXDESC_TYPE_COMMENT:
1461 case DMGBLKXDESC_TYPE_TERMINATOR:
[32432]1462 break;
1463 default:
[32536]1464 rc = VERR_VD_DMG_INVALID_HEADER;
[32432]1465 break;
1466 }
1467
[32768]1468 if ( pBlkxDesc->u32Type == DMGBLKXDESC_TYPE_TERMINATOR
[32432]1469 || RT_FAILURE(rc))
1470 break;
1471
1472 pBlkxDesc++;
1473 }
1474
1475 return rc;
1476}
1477
[48852]1478
[32432]1479/**
[48852]1480 * Worker for dmgOpenImage that tries to open a DMG inside a XAR file.
1481 *
1482 * We'll select the first .dmg inside the archive that we can get a file
1483 * interface to.
1484 *
1485 * @returns VBox status code.
1486 * @param fOpen Flags for defining the open type.
[48871]1487 * @param pVDIfIoInt The internal VD I/O interface to use.
[48852]1488 * @param pvStorage The storage pointer that goes with @a pVDIfsIo.
1489 * @param pszFilename The input filename, optional.
1490 * @param phXarFss Where to return the XAR file system stream handle on
1491 * success
1492 * @param phDmgFileInXar Where to return the VFS handle to the DMG file
1493 * within the XAR image on success.
1494 *
1495 * @remarks Not using the PDMGIMAGE structure directly here because the function
1496 * is being in serveral places.
1497 */
[48871]1498static int dmgOpenImageWithinXar(uint32_t fOpen, PVDINTERFACEIOINT pVDIfIoInt, void *pvStorage, const char *pszFilename,
[48852]1499 PRTVFSFSSTREAM phXarFss, PRTVFSFILE phDmgFileInXar)
1500{
1501 /*
1502 * Open the XAR file stream.
1503 */
1504 RTVFSFILE hVfsFile;
[48871]1505#ifdef VBOX_WITH_DIRECT_XAR_ACCESS
1506 int rc = RTVfsFileOpenNormal(pszFilename, fOpen, &hVfsFile);
1507#else
1508 int rc = VDIfCreateVfsFile(NULL, pVDIfIoInt, pvStorage, fOpen, &hVfsFile);
1509#endif
[48852]1510 if (RT_FAILURE(rc))
1511 return rc;
1512
1513 RTVFSIOSTREAM hVfsIos = RTVfsFileToIoStream(hVfsFile);
1514 RTVfsFileRelease(hVfsFile);
1515
1516 RTVFSFSSTREAM hXarFss;
1517 rc = RTZipXarFsStreamFromIoStream(hVfsIos, 0 /*fFlags*/, &hXarFss);
1518 RTVfsIoStrmRelease(hVfsIos);
1519 if (RT_FAILURE(rc))
1520 return rc;
1521
1522 /*
1523 * Look for a DMG in the stream that we can use.
1524 */
1525 for (;;)
1526 {
1527 char *pszName;
1528 RTVFSOBJTYPE enmType;
1529 RTVFSOBJ hVfsObj;
1530 rc = RTVfsFsStrmNext(hXarFss, &pszName, &enmType, &hVfsObj);
1531 if (RT_FAILURE(rc))
1532 break;
1533
1534 /* It must be a file object so it can be seeked, this also implies that
1535 it's uncompressed. Then it must have the .dmg suffix. */
1536 if (enmType == RTVFSOBJTYPE_FILE)
1537 {
1538 size_t cchName = strlen(pszName);
1539 const char *pszSuff = pszName + cchName - 4;
1540 if ( cchName >= 4
1541 && pszSuff[0] == '.'
1542 && (pszSuff[1] == 'd' || pszSuff[1] == 'D')
1543 && (pszSuff[2] == 'm' || pszSuff[2] == 'M')
1544 && (pszSuff[3] == 'g' || pszSuff[3] == 'G'))
1545 {
1546 RTVFSFILE hDmgFileInXar = RTVfsObjToFile(hVfsObj);
1547 AssertBreakStmt(hDmgFileInXar != NIL_RTVFSFILE, rc = VERR_INTERNAL_ERROR_3);
1548
1549 if (pszFilename)
1550 DMG_PRINTF(("DMG: Using '%s' within XAR file '%s'...\n", pszName, pszFilename));
1551 *phXarFss = hXarFss;
1552 *phDmgFileInXar = hDmgFileInXar;
1553
1554 RTStrFree(pszName);
1555 RTVfsObjRelease(hVfsObj);
1556
1557 return VINF_SUCCESS;
1558 }
1559 }
1560
1561 /* Release the current return values. */
1562 RTStrFree(pszName);
1563 RTVfsObjRelease(hVfsObj);
1564 }
1565
1566 /* Not found or some kind of error. */
1567 RTVfsFsStrmRelease(hXarFss);
1568 if (rc == VERR_EOF)
1569 rc = VERR_VD_DMG_NOT_FOUND_INSIDE_XAR;
1570 AssertStmt(RT_FAILURE_NP(rc), rc = VERR_INTERNAL_ERROR_4);
1571 return rc;
1572}
1573
1574
1575/**
[32536]1576 * Worker for dmgOpen that reads in and validates all the necessary
[21401]1577 * structures from the image.
1578 *
1579 * @returns VBox status code.
1580 * @param pThis The DMG instance data.
[32536]1581 * @param uOpenFlags Flags for defining the open type.
[21401]1582 */
[57388]1583static DECLCALLBACK(int) dmgOpenImage(PDMGIMAGE pThis, unsigned uOpenFlags)
[21401]1584{
[32536]1585 pThis->uOpenFlags = uOpenFlags;
1586
[38469]1587 pThis->pIfError = VDIfErrorGet(pThis->pVDIfsDisk);
[48852]1588 pThis->pIfIoXxx = VDIfIoIntGet(pThis->pVDIfsImage);
1589 pThis->hDmgFileInXar = NIL_RTVFSFILE;
1590 pThis->hXarFss = NIL_RTVFSFSSTREAM;
1591 AssertPtrReturn(pThis->pIfIoXxx, VERR_INVALID_PARAMETER);
[32536]1592
[48852]1593 int rc = vdIfIoIntFileOpen(pThis->pIfIoXxx, pThis->pszFilename,
1594 VDOpenFlagsToFileOpenFlags(uOpenFlags, false /* fCreate */),
[38469]1595 &pThis->pStorage);
[32536]1596 if (RT_FAILURE(rc))
1597 {
1598 /* Do NOT signal an appropriate error here, as the VD layer has the
1599 * choice of retrying the open if it failed. */
1600 return rc;
1601 }
1602
[21401]1603 /*
[48852]1604 * Check for XAR archive.
1605 */
1606 uint32_t u32XarMagic;
1607 rc = dmgWrapFileReadSync(pThis, 0, &u32XarMagic, sizeof(u32XarMagic));
1608 if (RT_FAILURE(rc))
1609 return rc;
1610 if (u32XarMagic == XAR_HEADER_MAGIC)
1611 {
1612 rc = dmgOpenImageWithinXar(VDOpenFlagsToFileOpenFlags(uOpenFlags, false /* fCreate */),
[48871]1613 pThis->pIfIoXxx,
[48852]1614 pThis->pStorage,
1615 pThis->pszFilename,
1616 &pThis->hXarFss, &pThis->hDmgFileInXar);
1617 if (RT_FAILURE(rc))
1618 return rc;
[48871]1619#ifdef VBOX_WITH_DIRECT_XAR_ACCESS
1620 vdIfIoIntFileClose(pThis->pIfIoXxx, pThis->pStorage);
1621 pThis->pStorage = NULL;
1622#endif
[48852]1623 }
[48871]1624#if 0 /* This is for testing whether the VFS wrappers actually works. */
1625 else
1626 {
1627 rc = RTVfsFileOpenNormal(pThis->pszFilename, VDOpenFlagsToFileOpenFlags(uOpenFlags, false /* fCreate */),
1628 &pThis->hDmgFileInXar);
1629 if (RT_FAILURE(rc))
1630 return rc;
1631 vdIfIoIntFileClose(pThis->pIfIoXxx, pThis->pStorage);
1632 pThis->pStorage = NULL;
1633 }
1634#endif
[48852]1635
1636 /*
[21401]1637 * Read the footer.
1638 */
[48852]1639 rc = dmgWrapFileGetSize(pThis, &pThis->cbFile);
[21401]1640 if (RT_FAILURE(rc))
1641 return rc;
1642 if (pThis->cbFile < 1024)
[32536]1643 return VERR_VD_DMG_INVALID_HEADER;
[48852]1644 rc = dmgWrapFileReadSync(pThis, pThis->cbFile - sizeof(pThis->Ftr), &pThis->Ftr, sizeof(pThis->Ftr));
[21401]1645 if (RT_FAILURE(rc))
1646 return rc;
[32768]1647 dmgUdifFtrFile2HostEndian(&pThis->Ftr);
[21401]1648
1649 /*
1650 * Do we recognize the footer structure? If so, is it valid?
1651 */
[32768]1652 if (pThis->Ftr.u32Magic != DMGUDIF_MAGIC)
[32536]1653 return VERR_VD_DMG_INVALID_HEADER;
[32768]1654 if (pThis->Ftr.u32Version != DMGUDIF_VER_CURRENT)
[32536]1655 return VERR_VD_DMG_INVALID_HEADER;
[21401]1656 if (pThis->Ftr.cbFooter != sizeof(pThis->Ftr))
[32536]1657 return VERR_VD_DMG_INVALID_HEADER;
[21401]1658
[32768]1659 if (!dmgUdifFtrIsValid(&pThis->Ftr, pThis->cbFile - sizeof(pThis->Ftr)))
[21401]1660 {
[32768]1661 DMG_PRINTF(("Bad DMG: '%s' cbFile=%RTfoff\n", pThis->pszFilename, pThis->cbFile));
[32536]1662 return VERR_VD_DMG_INVALID_HEADER;
[21401]1663 }
1664
[32768]1665 pThis->cbSize = pThis->Ftr.cSectors * DMG_SECTOR_SIZE;
[32432]1666
[21401]1667 /*
1668 * Read and parse the XML portion.
1669 */
1670 size_t cchXml = (size_t)pThis->Ftr.cbXml;
1671 char *pszXml = (char *)RTMemAlloc(cchXml + 1);
1672 if (!pszXml)
1673 return VERR_NO_MEMORY;
[48852]1674 rc = dmgWrapFileReadSync(pThis, pThis->Ftr.offXml, pszXml, cchXml);
[21401]1675 if (RT_SUCCESS(rc))
1676 {
1677 pszXml[cchXml] = '\0';
[32768]1678 const char *pszError = dmgOpenXmlToRsrc(pThis, pszXml);
[21401]1679 if (!pszError)
1680 {
[32768]1681 PCDMGUDIFRSRCARRAY pRsrcBlkx = NULL;
[32432]1682
1683 rc = dmgGetRsrcData(pThis, "blkx", &pRsrcBlkx);
1684 if (RT_SUCCESS(rc))
1685 {
1686 for (unsigned idxBlkx = 0; idxBlkx < pRsrcBlkx->cEntries; idxBlkx++)
1687 {
[32768]1688 PDMGBLKX pBlkx = NULL;
[32432]1689
[32768]1690 if (pRsrcBlkx->aEntries[idxBlkx].cbData < sizeof(DMGBLKX))
[32432]1691 {
[32536]1692 rc = VERR_VD_DMG_INVALID_HEADER;
[32432]1693 break;
1694 }
1695
[32768]1696 pBlkx = (PDMGBLKX)RTMemAllocZ(pRsrcBlkx->aEntries[idxBlkx].cbData);
[32432]1697 if (!pBlkx)
1698 {
1699 rc = VERR_NO_MEMORY;
1700 break;
1701 }
1702
1703 memcpy(pBlkx, pRsrcBlkx->aEntries[idxBlkx].pbData, pRsrcBlkx->aEntries[idxBlkx].cbData);
1704
[32768]1705 dmgBlkxFile2HostEndian(pBlkx);
[32432]1706
[32768]1707 if ( dmgBlkxIsValid(pBlkx)
1708 && pRsrcBlkx->aEntries[idxBlkx].cbData == pBlkx->cBlocksRunCount * sizeof(DMGBLKXDESC) + sizeof(DMGBLKX))
1709 rc = dmgBlkxParse(pThis, pBlkx);
[32432]1710 else
[32536]1711 rc = VERR_VD_DMG_INVALID_HEADER;
[32432]1712
[33479]1713 RTMemFree(pBlkx);
1714
[32432]1715 if (RT_FAILURE(rc))
1716 break;
1717 }
1718 }
1719 else
[32536]1720 rc = VERR_VD_DMG_INVALID_HEADER;
[21401]1721 }
1722 else
1723 {
[32768]1724 DMG_PRINTF(("**** XML DUMP BEGIN ***\n%s\n**** XML DUMP END ****\n", pszXml));
1725 DMG_PRINTF(("**** Bad XML at %#lx (%lu) ***\n%.256s\n**** Bad XML END ****\n",
[21401]1726 (unsigned long)(pszError - pszXml), (unsigned long)(pszError - pszXml), pszError));
1727 rc = VERR_VD_DMG_XML_PARSE_ERROR;
1728 }
1729 }
1730 RTMemFree(pszXml);
[32536]1731
[66486]1732 if (RT_SUCCESS(rc))
1733 {
1734 PVDREGIONDESC pRegion = &pThis->RegionList.aRegions[0];
1735 pThis->RegionList.fFlags = 0;
1736 pThis->RegionList.cRegions = 1;
1737
1738 pRegion->offRegion = 0; /* Disk start. */
1739 pRegion->cbBlock = 2048;
1740 pRegion->enmDataForm = VDREGIONDATAFORM_RAW;
1741 pRegion->enmMetadataForm = VDREGIONMETADATAFORM_NONE;
1742 pRegion->cbData = 2048;
1743 pRegion->cbMetadata = 0;
1744 pRegion->cRegionBlocksOrBytes = pThis->cbSize;
1745 }
1746 else
[32536]1747 dmgFreeImage(pThis, false);
1748 return rc;
1749}
[21401]1750
1751
[63802]1752/** @interface_method_impl{VDIMAGEBACKEND,pfnProbe} */
1753static DECLCALLBACK(int) dmgProbe(const char *pszFilename, PVDINTERFACE pVDIfsDisk,
[79965]1754 PVDINTERFACE pVDIfsImage, VDTYPE enmDesiredType, VDTYPE *penmType)
[32536]1755{
[79965]1756 RT_NOREF(pVDIfsDisk, enmDesiredType);
[33524]1757 LogFlowFunc(("pszFilename=\"%s\" pVDIfsDisk=%#p pVDIfsImage=%#p penmType=%#p\n",
1758 pszFilename, pVDIfsDisk, pVDIfsImage, penmType));
[32536]1759
[38469]1760 PVDINTERFACEIOINT pIfIo = VDIfIoIntGet(pVDIfsImage);
1761 AssertPtrReturn(pIfIo, VERR_INVALID_PARAMETER);
[32536]1762
1763 /*
[48852]1764 * Open the file and check for XAR.
[32536]1765 */
[48852]1766 PVDIOSTORAGE pStorage = NULL;
1767 int rc = vdIfIoIntFileOpen(pIfIo, pszFilename,
1768 VDOpenFlagsToFileOpenFlags(VD_OPEN_FLAGS_READONLY, false /* fCreate */),
1769 &pStorage);
1770 if (RT_FAILURE(rc))
[32536]1771 {
[48852]1772 LogFlowFunc(("returns %Rrc (error opening file)\n", rc));
1773 return rc;
[32536]1774 }
[48852]1775
1776 /*
1777 * Check for XAR file.
1778 */
1779 RTVFSFSSTREAM hXarFss = NIL_RTVFSFSSTREAM;
1780 RTVFSFILE hDmgFileInXar = NIL_RTVFSFILE;
1781 uint32_t u32XarMagic;
1782 rc = vdIfIoIntFileReadSync(pIfIo, pStorage, 0, &u32XarMagic, sizeof(u32XarMagic));
1783 if ( RT_SUCCESS(rc)
1784 && u32XarMagic == XAR_HEADER_MAGIC)
1785 {
1786 rc = dmgOpenImageWithinXar(RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE,
[48871]1787 pIfIo, pStorage, pszFilename,
[48852]1788 &hXarFss, &hDmgFileInXar);
1789 if (RT_FAILURE(rc))
1790 return rc;
1791 }
1792
1793 /*
1794 * Read the DMG footer.
1795 */
1796 uint64_t cbFile;
1797 if (hDmgFileInXar == NIL_RTVFSFILE)
1798 rc = vdIfIoIntFileGetSize(pIfIo, pStorage, &cbFile);
[32536]1799 else
[80585]1800 rc = RTVfsFileQuerySize(hDmgFileInXar, &cbFile);
[66212]1801 if ( RT_SUCCESS(rc)
1802 && cbFile >= sizeof(DMGUDIF))
[32536]1803 {
[48852]1804 DMGUDIF Ftr;
1805 uint64_t offFtr = cbFile - sizeof(Ftr);
1806 if (hDmgFileInXar == NIL_RTVFSFILE)
1807 rc = vdIfIoIntFileReadSync(pIfIo, pStorage, offFtr, &Ftr, sizeof(Ftr));
1808 else
1809 rc = RTVfsFileReadAt(hDmgFileInXar, offFtr, &Ftr, sizeof(Ftr), NULL);
1810 if (RT_SUCCESS(rc))
[32536]1811 {
[48852]1812 /*
1813 * Do we recognize this stuff? Does it look valid?
1814 */
[48871]1815 if ( Ftr.u32Magic == RT_H2BE_U32_C(DMGUDIF_MAGIC)
1816 && Ftr.u32Version == RT_H2BE_U32_C(DMGUDIF_VER_CURRENT)
1817 && Ftr.cbFooter == RT_H2BE_U32_C(sizeof(Ftr)))
[33524]1818 {
[48852]1819 dmgUdifFtrFile2HostEndian(&Ftr);
1820 if (dmgUdifFtrIsValid(&Ftr, offFtr))
1821 {
1822 rc = VINF_SUCCESS;
[66211]1823 *penmType = VDTYPE_OPTICAL_DISC;
[48852]1824 }
1825 else
1826 {
1827 DMG_PRINTF(("Bad DMG: '%s' offFtr=%RTfoff\n", pszFilename, offFtr));
1828 rc = VERR_VD_DMG_INVALID_HEADER;
1829 }
[33524]1830 }
[32536]1831 else
1832 rc = VERR_VD_DMG_INVALID_HEADER;
1833 }
[66212]1834 else
1835 rc = VERR_VD_DMG_INVALID_HEADER;
[32536]1836 }
[48852]1837 else
1838 rc = VERR_VD_DMG_INVALID_HEADER;
[32536]1839
[48852]1840 /* Clean up. */
1841 RTVfsFileRelease(hDmgFileInXar);
1842 RTVfsFsStrmRelease(hXarFss);
1843 vdIfIoIntFileClose(pIfIo, pStorage);
[32536]1844
1845 LogFlowFunc(("returns %Rrc\n", rc));
1846 return rc;
[21401]1847}
1848
[63785]1849/** @interface_method_impl{VDIMAGEBACKEND,pfnOpen} */
[48840]1850static DECLCALLBACK(int) dmgOpen(const char *pszFilename, unsigned uOpenFlags,
1851 PVDINTERFACE pVDIfsDisk, PVDINTERFACE pVDIfsImage,
1852 VDTYPE enmType, void **ppBackendData)
[21401]1853{
[54430]1854 LogFlowFunc(("pszFilename=\"%s\" uOpenFlags=%#x pVDIfsDisk=%#p pVDIfsImage=%#p enmType=%u ppBackendData=%#p\n", pszFilename, uOpenFlags, pVDIfsDisk, pVDIfsImage, enmType, ppBackendData));
[32536]1855
[54430]1856 NOREF(enmType); /**< @todo r=klaus make use of the type info. */
1857
[32536]1858 /* Check open flags. All valid flags are (in principle) supported. */
[48852]1859 AssertReturn(!(uOpenFlags & ~VD_OPEN_FLAGS_MASK), VERR_INVALID_PARAMETER);
[32536]1860
1861 /* Check remaining arguments. */
[48852]1862 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
1863 AssertReturn(*pszFilename, VERR_INVALID_PARAMETER);
[32536]1864
[21401]1865 /*
1866 * Reject combinations we don't currently support.
1867 *
1868 * There is no point in being paranoid about the input here as we're just a
1869 * simple backend and can expect the caller to be the only user and already
1870 * have validate what it passes thru to us.
1871 */
[36513]1872 if ( !(uOpenFlags & VD_OPEN_FLAGS_READONLY)
1873 || (uOpenFlags & VD_OPEN_FLAGS_ASYNC_IO))
[32536]1874 {
[48852]1875 LogFlowFunc(("Unsupported flag(s): %#x\n", uOpenFlags));
1876 return VERR_INVALID_PARAMETER;
[32536]1877 }
[21401]1878
1879 /*
1880 * Create the basic instance data structure and open the file,
1881 * then hand it over to a worker function that does all the rest.
1882 */
[48852]1883 int rc = VERR_NO_MEMORY;
[66486]1884 PDMGIMAGE pThis = (PDMGIMAGE)RTMemAllocZ(RT_UOFFSETOF(DMGIMAGE, RegionList.aRegions[1]));
[48852]1885 if (pThis)
[32536]1886 {
[48852]1887 pThis->pszFilename = pszFilename;
1888 pThis->pStorage = NULL;
1889 pThis->pVDIfsDisk = pVDIfsDisk;
1890 pThis->pVDIfsImage = pVDIfsImage;
1891
1892 rc = dmgOpenImage(pThis, uOpenFlags);
1893 if (RT_SUCCESS(rc))
1894 *ppBackendData = pThis;
1895 else
1896 RTMemFree(pThis);
[32536]1897 }
1898
1899 LogFlowFunc(("returns %Rrc (pBackendData=%#p)\n", rc, *ppBackendData));
[21401]1900 return rc;
1901}
1902
[63785]1903/** @interface_method_impl{VDIMAGEBACKEND,pfnCreate} */
[48840]1904static DECLCALLBACK(int) dmgCreate(const char *pszFilename, uint64_t cbSize,
1905 unsigned uImageFlags, const char *pszComment,
1906 PCVDGEOMETRY pPCHSGeometry, PCVDGEOMETRY pLCHSGeometry,
1907 PCRTUUID pUuid, unsigned uOpenFlags,
1908 unsigned uPercentStart, unsigned uPercentSpan,
1909 PVDINTERFACE pVDIfsDisk, PVDINTERFACE pVDIfsImage,
[54430]1910 PVDINTERFACE pVDIfsOperation, VDTYPE enmType,
1911 void **ppBackendData)
[32432]1912{
[62752]1913 RT_NOREF8(pszFilename, cbSize, uImageFlags, pszComment, pPCHSGeometry, pLCHSGeometry, pUuid, uOpenFlags);
1914 RT_NOREF7(uPercentStart, uPercentSpan, pVDIfsDisk, pVDIfsImage, pVDIfsOperation, enmType, ppBackendData);
[54430]1915 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",
1916 pszFilename, cbSize, uImageFlags, pszComment, pPCHSGeometry, pLCHSGeometry, pUuid, uOpenFlags, uPercentStart, uPercentSpan, pVDIfsDisk, pVDIfsImage, pVDIfsOperation, enmType, ppBackendData));
[32536]1917 int rc = VERR_NOT_SUPPORTED;
1918
1919 LogFlowFunc(("returns %Rrc\n", rc));
1920 return rc;
[32432]1921}
[21401]1922
[63785]1923/** @interface_method_impl{VDIMAGEBACKEND,pfnRename} */
[57388]1924static DECLCALLBACK(int) dmgRename(void *pBackendData, const char *pszFilename)
[32432]1925{
[62752]1926 RT_NOREF2(pBackendData, pszFilename);
[32432]1927 LogFlowFunc(("pBackendData=%#p pszFilename=%#p\n", pBackendData, pszFilename));
1928 int rc = VERR_NOT_SUPPORTED;
[21401]1929
[32432]1930 LogFlowFunc(("returns %Rrc\n", rc));
1931 return rc;
1932}
[21401]1933
[63785]1934/** @interface_method_impl{VDIMAGEBACKEND,pfnClose} */
[48840]1935static DECLCALLBACK(int) dmgClose(void *pBackendData, bool fDelete)
[21401]1936{
[32432]1937 LogFlowFunc(("pBackendData=%#p fDelete=%d\n", pBackendData, fDelete));
[32536]1938 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[21401]1939
[48840]1940 int rc = dmgFreeImage(pThis, fDelete);
[32536]1941 RTMemFree(pThis);
[32432]1942
1943 LogFlowFunc(("returns %Rrc\n", rc));
1944 return rc;
1945}
1946
[63785]1947/** @interface_method_impl{VDIMAGEBACKEND,pfnRead} */
[48840]1948static DECLCALLBACK(int) dmgRead(void *pBackendData, uint64_t uOffset, size_t cbToRead,
1949 PVDIOCTX pIoCtx, size_t *pcbActuallyRead)
[32432]1950{
[44252]1951 LogFlowFunc(("pBackendData=%#p uOffset=%llu pIoCtx=%#p cbToRead=%zu pcbActuallyRead=%#p\n",
1952 pBackendData, uOffset, pIoCtx, cbToRead, pcbActuallyRead));
[32536]1953 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[32432]1954 PDMGEXTENT pExtent = NULL;
[32500]1955 int rc = VINF_SUCCESS;
[32432]1956
[32536]1957 AssertPtr(pThis);
[32768]1958 Assert(uOffset % DMG_SECTOR_SIZE == 0);
1959 Assert(cbToRead % DMG_SECTOR_SIZE == 0);
[32432]1960
[32536]1961 if ( uOffset + cbToRead > pThis->cbSize
[32432]1962 || cbToRead == 0)
[21401]1963 {
[48840]1964 LogFlowFunc(("returns VERR_INVALID_PARAMETER\n"));
1965 return VERR_INVALID_PARAMETER;
[21401]1966 }
1967
[32768]1968 pExtent = dmgExtentGetFromOffset(pThis, DMG_BYTE2BLOCK(uOffset));
[21401]1969
[32432]1970 if (pExtent)
[21401]1971 {
[32768]1972 uint64_t uExtentRel = DMG_BYTE2BLOCK(uOffset) - pExtent->uSectorExtent;
[32432]1973
1974 /* Remain in this extent. */
[32768]1975 cbToRead = RT_MIN(cbToRead, DMG_BLOCK2BYTE(pExtent->cSectorsExtent - uExtentRel));
[32432]1976
1977 switch (pExtent->enmType)
1978 {
1979 case DMGEXTENTTYPE_RAW:
1980 {
[48852]1981 rc = dmgWrapFileReadUser(pThis, pExtent->offFileStart + DMG_BLOCK2BYTE(uExtentRel), pIoCtx, cbToRead);
[32432]1982 break;
1983 }
1984 case DMGEXTENTTYPE_ZERO:
1985 {
[48852]1986 vdIfIoIntIoCtxSet(pThis->pIfIoXxx, pIoCtx, 0, cbToRead);
[32432]1987 break;
1988 }
[32768]1989 case DMGEXTENTTYPE_COMP_ZLIB:
1990 {
1991 if (pThis->pExtentDecomp != pExtent)
1992 {
1993 if (DMG_BLOCK2BYTE(pExtent->cSectorsExtent) > pThis->cbDecompExtent)
1994 {
1995 if (RT_LIKELY(pThis->pvDecompExtent))
1996 RTMemFree(pThis->pvDecompExtent);
1997
1998 pThis->pvDecompExtent = RTMemAllocZ(DMG_BLOCK2BYTE(pExtent->cSectorsExtent));
1999 if (!pThis->pvDecompExtent)
2000 rc = VERR_NO_MEMORY;
2001 else
2002 pThis->cbDecompExtent = DMG_BLOCK2BYTE(pExtent->cSectorsExtent);
2003 }
2004
2005 if (RT_SUCCESS(rc))
2006 {
2007 rc = dmgFileInflateSync(pThis, pExtent->offFileStart, pExtent->cbFile,
2008 pThis->pvDecompExtent,
2009 RT_MIN(pThis->cbDecompExtent, DMG_BLOCK2BYTE(pExtent->cSectorsExtent)));
2010 if (RT_SUCCESS(rc))
2011 pThis->pExtentDecomp = pExtent;
2012 }
2013 }
2014
2015 if (RT_SUCCESS(rc))
[48852]2016 vdIfIoIntIoCtxCopyTo(pThis->pIfIoXxx, pIoCtx,
[44252]2017 (uint8_t *)pThis->pvDecompExtent + DMG_BLOCK2BYTE(uExtentRel),
2018 cbToRead);
[32768]2019 break;
2020 }
[32432]2021 default:
2022 AssertMsgFailed(("Invalid extent type\n"));
2023 }
2024
2025 if (RT_SUCCESS(rc))
2026 *pcbActuallyRead = cbToRead;
[21401]2027 }
[32432]2028 else
2029 rc = VERR_INVALID_PARAMETER;
[21401]2030
[32432]2031 LogFlowFunc(("returns %Rrc\n", rc));
2032 return rc;
2033}
[21401]2034
[63785]2035/** @interface_method_impl{VDIMAGEBACKEND,pfnWrite} */
[48840]2036static DECLCALLBACK(int) dmgWrite(void *pBackendData, uint64_t uOffset, size_t cbToWrite,
2037 PVDIOCTX pIoCtx, size_t *pcbWriteProcess, size_t *pcbPreRead,
2038 size_t *pcbPostRead, unsigned fWrite)
[32432]2039{
[62752]2040 RT_NOREF7(uOffset, cbToWrite, pIoCtx, pcbWriteProcess, pcbPreRead, pcbPostRead, fWrite);
[44252]2041 LogFlowFunc(("pBackendData=%#p uOffset=%llu pIoCtx=%#p cbToWrite=%zu pcbWriteProcess=%#p pcbPreRead=%#p pcbPostRead=%#p\n",
2042 pBackendData, uOffset, pIoCtx, cbToWrite, pcbWriteProcess, pcbPreRead, pcbPostRead));
[32536]2043 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[32500]2044 int rc = VERR_NOT_IMPLEMENTED;
[32432]2045
[32536]2046 AssertPtr(pThis);
[32432]2047 Assert(uOffset % 512 == 0);
2048 Assert(cbToWrite % 512 == 0);
2049
[48840]2050 if (!(pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY))
2051 AssertMsgFailed(("Not implemented\n"));
2052 else
[32432]2053 rc = VERR_VD_IMAGE_READ_ONLY;
2054
2055 LogFlowFunc(("returns %Rrc\n", rc));
2056 return rc;
2057}
2058
[63785]2059/** @interface_method_impl{VDIMAGEBACKEND,pfnFlush} */
[48840]2060static DECLCALLBACK(int) dmgFlush(void *pBackendData, PVDIOCTX pIoCtx)
[32432]2061{
[62752]2062 RT_NOREF1(pIoCtx);
[32432]2063 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
[32536]2064 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
2065 int rc;
[32432]2066
[32536]2067 AssertPtr(pThis);
2068
2069 rc = dmgFlushImage(pThis);
2070
[32432]2071 LogFlowFunc(("returns %Rrc\n", rc));
2072 return rc;
2073}
2074
[63785]2075/** @interface_method_impl{VDIMAGEBACKEND,pfnGetVersion} */
[48840]2076static DECLCALLBACK(unsigned) dmgGetVersion(void *pBackendData)
[32432]2077{
2078 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
[32536]2079 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[32432]2080
[63786]2081 AssertPtrReturn(pThis, 0);
[32432]2082
[63786]2083 return 1;
[32432]2084}
2085
[63785]2086/** @interface_method_impl{VDIMAGEBACKEND,pfnGetFileSize} */
[48840]2087static DECLCALLBACK(uint64_t) dmgGetFileSize(void *pBackendData)
[32432]2088{
2089 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
[32536]2090 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[32432]2091
[63786]2092 AssertPtrReturn(pThis, 0);
[32432]2093
[63786]2094 uint64_t cbFile = 0;
2095 if (pThis->pStorage || pThis->hDmgFileInXar != NIL_RTVFSFILE)
[32432]2096 {
[48852]2097 int rc = dmgWrapFileGetSize(pThis, &cbFile);
[63786]2098 if (RT_FAILURE(rc))
2099 cbFile = 0; /* Make sure it is 0 */
[21401]2100 }
2101
[63786]2102 LogFlowFunc(("returns %lld\n", cbFile));
2103 return cbFile;
[32432]2104}
2105
[63785]2106/** @interface_method_impl{VDIMAGEBACKEND,pfnGetPCHSGeometry} */
[48840]2107static DECLCALLBACK(int) dmgGetPCHSGeometry(void *pBackendData, PVDGEOMETRY pPCHSGeometry)
[32432]2108{
2109 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p\n", pBackendData, pPCHSGeometry));
[32536]2110 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[63786]2111 int rc = VINF_SUCCESS;
[32432]2112
[63786]2113 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
[32432]2114
[63786]2115 if (pThis->PCHSGeometry.cCylinders)
2116 *pPCHSGeometry = pThis->PCHSGeometry;
[32432]2117 else
[63786]2118 rc = VERR_VD_GEOMETRY_NOT_SET;
[21401]2119
[32432]2120 LogFlowFunc(("returns %Rrc (PCHS=%u/%u/%u)\n", rc, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
2121 return rc;
2122}
[21401]2123
[63785]2124/** @interface_method_impl{VDIMAGEBACKEND,pfnSetPCHSGeometry} */
[48840]2125static DECLCALLBACK(int) dmgSetPCHSGeometry(void *pBackendData, PCVDGEOMETRY pPCHSGeometry)
[32432]2126{
2127 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p PCHS=%u/%u/%u\n",
2128 pBackendData, pPCHSGeometry, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
[32536]2129 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[63786]2130 int rc = VINF_SUCCESS;
[32432]2131
[63786]2132 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
[32432]2133
[63786]2134 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2135 rc = VERR_VD_IMAGE_READ_ONLY;
[32432]2136 else
[63786]2137 pThis->PCHSGeometry = *pPCHSGeometry;
[32432]2138
2139 LogFlowFunc(("returns %Rrc\n", rc));
2140 return rc;
[21401]2141}
[32432]2142
[63785]2143/** @interface_method_impl{VDIMAGEBACKEND,pfnGetLCHSGeometry} */
[48840]2144static DECLCALLBACK(int) dmgGetLCHSGeometry(void *pBackendData, PVDGEOMETRY pLCHSGeometry)
[32432]2145{
[63786]2146 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p\n", pBackendData, pLCHSGeometry));
[32536]2147 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[63786]2148 int rc = VINF_SUCCESS;
[32432]2149
[63786]2150 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
[32432]2151
[63786]2152 if (pThis->LCHSGeometry.cCylinders)
2153 *pLCHSGeometry = pThis->LCHSGeometry;
[32432]2154 else
[63786]2155 rc = VERR_VD_GEOMETRY_NOT_SET;
[32432]2156
2157 LogFlowFunc(("returns %Rrc (LCHS=%u/%u/%u)\n", rc, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
2158 return rc;
2159}
2160
[63785]2161/** @interface_method_impl{VDIMAGEBACKEND,pfnSetLCHSGeometry} */
[48840]2162static DECLCALLBACK(int) dmgSetLCHSGeometry(void *pBackendData, PCVDGEOMETRY pLCHSGeometry)
[32432]2163{
2164 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p LCHS=%u/%u/%u\n",
2165 pBackendData, pLCHSGeometry, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
[32536]2166 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[63786]2167 int rc = VINF_SUCCESS;
[32432]2168
[63786]2169 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
[32432]2170
[63786]2171 if (pThis->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2172 rc = VERR_VD_IMAGE_READ_ONLY;
[32432]2173 else
[63786]2174 pThis->LCHSGeometry = *pLCHSGeometry;
[32432]2175
2176 LogFlowFunc(("returns %Rrc\n", rc));
2177 return rc;
2178}
2179
[66486]2180/** @copydoc VDIMAGEBACKEND::pfnQueryRegions */
2181static DECLCALLBACK(int) dmgQueryRegions(void *pBackendData, PCVDREGIONLIST *ppRegionList)
2182{
2183 LogFlowFunc(("pBackendData=%#p ppRegionList=%#p\n", pBackendData, ppRegionList));
2184 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
2185
2186 AssertPtrReturn(pThis, VERR_VD_NOT_OPENED);
2187
2188 *ppRegionList = &pThis->RegionList;
2189 LogFlowFunc(("returns %Rrc\n", VINF_SUCCESS));
2190 return VINF_SUCCESS;
2191}
2192
2193/** @copydoc VDIMAGEBACKEND::pfnRegionListRelease */
2194static DECLCALLBACK(void) dmgRegionListRelease(void *pBackendData, PCVDREGIONLIST pRegionList)
2195{
2196 RT_NOREF1(pRegionList);
2197 LogFlowFunc(("pBackendData=%#p pRegionList=%#p\n", pBackendData, pRegionList));
2198 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
2199 AssertPtr(pThis); RT_NOREF(pThis);
2200
2201 /* Nothing to do here. */
2202}
2203
[63785]2204/** @interface_method_impl{VDIMAGEBACKEND,pfnGetImageFlags} */
[48840]2205static DECLCALLBACK(unsigned) dmgGetImageFlags(void *pBackendData)
[32432]2206{
2207 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
[32536]2208 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[63786]2209 AssertPtrReturn(pThis, 0);
[32432]2210
[63786]2211 LogFlowFunc(("returns %#x\n", pThis->uImageFlags));
2212 return pThis->uImageFlags;
[32432]2213}
2214
[63785]2215/** @interface_method_impl{VDIMAGEBACKEND,pfnGetOpenFlags} */
[48840]2216static DECLCALLBACK(unsigned) dmgGetOpenFlags(void *pBackendData)
[32432]2217{
2218 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
[32536]2219 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[32432]2220
[63786]2221 AssertPtrReturn(pThis, 0);
[32432]2222
[63786]2223 LogFlowFunc(("returns %#x\n", pThis->uOpenFlags));
2224 return pThis->uOpenFlags;
[32432]2225}
2226
[63785]2227/** @interface_method_impl{VDIMAGEBACKEND,pfnSetOpenFlags} */
[48840]2228static DECLCALLBACK(int) dmgSetOpenFlags(void *pBackendData, unsigned uOpenFlags)
[32432]2229{
2230 LogFlowFunc(("pBackendData=%#p\n uOpenFlags=%#x", pBackendData, uOpenFlags));
[32536]2231 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[63786]2232 int rc = VINF_SUCCESS;
[32432]2233
[33182]2234 /* Image must be opened and the new flags must be valid. */
[44232]2235 if (!pThis || (uOpenFlags & ~( VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO
2236 | VD_OPEN_FLAGS_SHAREABLE | VD_OPEN_FLAGS_SEQUENTIAL
2237 | VD_OPEN_FLAGS_SKIP_CONSISTENCY_CHECKS)))
[63786]2238 rc = VERR_INVALID_PARAMETER;
2239 else
[32432]2240 {
[63786]2241 /* Implement this operation via reopening the image. */
2242 rc = dmgFreeImage(pThis, false);
2243 if (RT_SUCCESS(rc))
2244 rc = dmgOpenImage(pThis, uOpenFlags);
[32432]2245 }
2246
2247 LogFlowFunc(("returns %Rrc\n", rc));
2248 return rc;
2249}
2250
[66494]2251/** @copydoc VDIMAGEBACKEND::pfnGetComment */
2252VD_BACKEND_CALLBACK_GET_COMMENT_DEF_NOT_SUPPORTED(dmgGetComment);
[32432]2253
[66494]2254/** @copydoc VDIMAGEBACKEND::pfnSetComment */
2255VD_BACKEND_CALLBACK_SET_COMMENT_DEF_NOT_SUPPORTED(dmgSetComment, PDMGIMAGE);
[32432]2256
[66494]2257/** @copydoc VDIMAGEBACKEND::pfnGetUuid */
2258VD_BACKEND_CALLBACK_GET_UUID_DEF_NOT_SUPPORTED(dmgGetUuid);
[32432]2259
[66494]2260/** @copydoc VDIMAGEBACKEND::pfnSetUuid */
2261VD_BACKEND_CALLBACK_SET_UUID_DEF_NOT_SUPPORTED(dmgSetUuid, PDMGIMAGE);
[32432]2262
[66494]2263/** @copydoc VDIMAGEBACKEND::pfnGetModificationUuid */
2264VD_BACKEND_CALLBACK_GET_UUID_DEF_NOT_SUPPORTED(dmgGetModificationUuid);
[32432]2265
[66494]2266/** @copydoc VDIMAGEBACKEND::pfnSetModificationUuid */
2267VD_BACKEND_CALLBACK_SET_UUID_DEF_NOT_SUPPORTED(dmgSetModificationUuid, PDMGIMAGE);
[32432]2268
[66494]2269/** @copydoc VDIMAGEBACKEND::pfnGetParentUuid */
2270VD_BACKEND_CALLBACK_GET_UUID_DEF_NOT_SUPPORTED(dmgGetParentUuid);
[32432]2271
[66494]2272/** @copydoc VDIMAGEBACKEND::pfnSetParentUuid */
2273VD_BACKEND_CALLBACK_SET_UUID_DEF_NOT_SUPPORTED(dmgSetParentUuid, PDMGIMAGE);
[32432]2274
[66494]2275/** @copydoc VDIMAGEBACKEND::pfnGetParentModificationUuid */
2276VD_BACKEND_CALLBACK_GET_UUID_DEF_NOT_SUPPORTED(dmgGetParentModificationUuid);
[32432]2277
[66494]2278/** @copydoc VDIMAGEBACKEND::pfnSetParentModificationUuid */
2279VD_BACKEND_CALLBACK_SET_UUID_DEF_NOT_SUPPORTED(dmgSetParentModificationUuid, PDMGIMAGE);
[32432]2280
[63785]2281/** @interface_method_impl{VDIMAGEBACKEND,pfnDump} */
[48840]2282static DECLCALLBACK(void) dmgDump(void *pBackendData)
[32432]2283{
[32536]2284 PDMGIMAGE pThis = (PDMGIMAGE)pBackendData;
[32432]2285
[63786]2286 AssertPtrReturnVoid(pThis);
2287 vdIfErrorMessage(pThis->pIfError, "Header: Geometry PCHS=%u/%u/%u LCHS=%u/%u/%u cSectors=%llu\n",
2288 pThis->PCHSGeometry.cCylinders, pThis->PCHSGeometry.cHeads, pThis->PCHSGeometry.cSectors,
2289 pThis->LCHSGeometry.cCylinders, pThis->LCHSGeometry.cHeads, pThis->LCHSGeometry.cSectors,
2290 pThis->cbSize / DMG_SECTOR_SIZE);
[32432]2291}
2292
2293
[63781]2294const VDIMAGEBACKEND g_DmgBackend =
[32432]2295{
[63905]2296 /* u32Version */
2297 VD_IMGBACKEND_VERSION,
[32432]2298 /* pszBackendName */
2299 "DMG",
2300 /* uBackendCaps */
[32536]2301 VD_CAP_FILE | VD_CAP_VFS,
[33524]2302 /* paFileExtensions */
2303 s_aDmgFileExtensions,
[32432]2304 /* paConfigInfo */
2305 NULL,
[63802]2306 /* pfnProbe */
2307 dmgProbe,
[32432]2308 /* pfnOpen */
2309 dmgOpen,
2310 /* pfnCreate */
2311 dmgCreate,
2312 /* pfnRename */
2313 dmgRename,
2314 /* pfnClose */
2315 dmgClose,
2316 /* pfnRead */
2317 dmgRead,
2318 /* pfnWrite */
2319 dmgWrite,
2320 /* pfnFlush */
2321 dmgFlush,
[44252]2322 /* pfnDiscard */
2323 NULL,
[32432]2324 /* pfnGetVersion */
2325 dmgGetVersion,
2326 /* pfnGetFileSize */
2327 dmgGetFileSize,
2328 /* pfnGetPCHSGeometry */
2329 dmgGetPCHSGeometry,
2330 /* pfnSetPCHSGeometry */
2331 dmgSetPCHSGeometry,
2332 /* pfnGetLCHSGeometry */
2333 dmgGetLCHSGeometry,
2334 /* pfnSetLCHSGeometry */
2335 dmgSetLCHSGeometry,
[66110]2336 /* pfnQueryRegions */
[66486]2337 dmgQueryRegions,
[66110]2338 /* pfnRegionListRelease */
[66486]2339 dmgRegionListRelease,
[32432]2340 /* pfnGetImageFlags */
2341 dmgGetImageFlags,
2342 /* pfnGetOpenFlags */
2343 dmgGetOpenFlags,
2344 /* pfnSetOpenFlags */
2345 dmgSetOpenFlags,
2346 /* pfnGetComment */
2347 dmgGetComment,
2348 /* pfnSetComment */
2349 dmgSetComment,
2350 /* pfnGetUuid */
2351 dmgGetUuid,
2352 /* pfnSetUuid */
2353 dmgSetUuid,
2354 /* pfnGetModificationUuid */
2355 dmgGetModificationUuid,
2356 /* pfnSetModificationUuid */
2357 dmgSetModificationUuid,
2358 /* pfnGetParentUuid */
2359 dmgGetParentUuid,
2360 /* pfnSetParentUuid */
2361 dmgSetParentUuid,
2362 /* pfnGetParentModificationUuid */
2363 dmgGetParentModificationUuid,
2364 /* pfnSetParentModificationUuid */
2365 dmgSetParentModificationUuid,
2366 /* pfnDump */
2367 dmgDump,
[58132]2368 /* pfnGetTimestamp */
[32536]2369 NULL,
[58132]2370 /* pfnGetParentTimestamp */
[32536]2371 NULL,
[58132]2372 /* pfnSetParentTimestamp */
[32536]2373 NULL,
[32432]2374 /* pfnGetParentFilename */
[32536]2375 NULL,
[32432]2376 /* pfnSetParentFilename */
[32536]2377 NULL,
[32432]2378 /* pfnComposeLocation */
2379 genericFileComposeLocation,
2380 /* pfnComposeName */
2381 genericFileComposeName,
2382 /* pfnCompact */
2383 NULL,
2384 /* pfnResize */
[38621]2385 NULL,
[39519]2386 /* pfnRepair */
[50988]2387 NULL,
2388 /* pfnTraverseMetadata */
[63905]2389 NULL,
2390 /* u32VersionEnd */
2391 VD_IMGBACKEND_VERSION
[32432]2392};
[48839]2393
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