VirtualBox

source: vbox/trunk/src/VBox/Storage/VMDK.cpp@ 99739

Last change on this file since 99739 was 99739, checked in by vboxsync, 16 months ago

*: doxygen corrections (mostly about removing @returns from functions returning void).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 379.2 KB
Line 
1/* $Id: VMDK.cpp 99739 2023-05-11 01:01:08Z vboxsync $ */
2/** @file
3 * VMDK disk image, core code.
4 */
5
6/*
7 * Copyright (C) 2006-2023 Oracle and/or its affiliates.
8 *
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
26 */
27
28
29/*********************************************************************************************************************************
30* Header Files *
31*********************************************************************************************************************************/
32#define LOG_GROUP LOG_GROUP_VD_VMDK
33#include <VBox/log.h> /* before VBox/vd-ifs.h */
34#include <VBox/vd-plugin.h>
35#include <VBox/err.h>
36
37#include <iprt/assert.h>
38#include <iprt/alloc.h>
39#include <iprt/base64.h>
40#include <iprt/ctype.h>
41#include <iprt/crc.h>
42#include <iprt/dvm.h>
43#include <iprt/uuid.h>
44#include <iprt/path.h>
45#include <iprt/rand.h>
46#include <iprt/string.h>
47#include <iprt/sort.h>
48#include <iprt/zip.h>
49#include <iprt/asm.h>
50#ifdef RT_OS_WINDOWS
51# include <iprt/utf16.h>
52# include <iprt/uni.h>
53# include <iprt/uni.h>
54# include <iprt/nt/nt-and-windows.h>
55# include <winioctl.h>
56#endif
57#ifdef RT_OS_LINUX
58# include <errno.h>
59# include <sys/stat.h>
60# include <iprt/dir.h>
61# include <iprt/symlink.h>
62# include <iprt/linux/sysfs.h>
63#endif
64#ifdef RT_OS_FREEBSD
65#include <libgeom.h>
66#include <sys/stat.h>
67#include <stdlib.h>
68#endif
69#ifdef RT_OS_SOLARIS
70#include <sys/dkio.h>
71#include <sys/vtoc.h>
72#include <sys/efi_partition.h>
73#include <unistd.h>
74#include <errno.h>
75#endif
76#ifdef RT_OS_DARWIN
77# include <sys/stat.h>
78# include <sys/disk.h>
79# include <errno.h>
80/* The following structure and IOCTLs are defined in znu bsd/sys/disk.h but
81 inside KERNEL ifdefs and thus stripped from the SDK edition of the header.
82 While we could try include the header from the Kernel.framework, it's a lot
83 easier to just add the structure and 4 defines here. */
84typedef struct
85{
86 uint64_t offset;
87 uint64_t length;
88 uint8_t reserved0128[12];
89 dev_t dev;
90} dk_physical_extent_t;
91# define DKIOCGETBASE _IOR( 'd', 73, uint64_t)
92# define DKIOCLOCKPHYSICALEXTENTS _IO( 'd', 81)
93# define DKIOCGETPHYSICALEXTENT _IOWR('d', 82, dk_physical_extent_t)
94# define DKIOCUNLOCKPHYSICALEXTENTS _IO( 'd', 83)
95#endif /* RT_OS_DARWIN */
96
97#include "VDBackends.h"
98
99
100/*********************************************************************************************************************************
101* Constants And Macros, Structures and Typedefs *
102*********************************************************************************************************************************/
103
104/** Maximum encoded string size (including NUL) we allow for VMDK images.
105 * Deliberately not set high to avoid running out of descriptor space. */
106#define VMDK_ENCODED_COMMENT_MAX 1024
107
108/** VMDK descriptor DDB entry for PCHS cylinders. */
109#define VMDK_DDB_GEO_PCHS_CYLINDERS "ddb.geometry.cylinders"
110
111/** VMDK descriptor DDB entry for PCHS heads. */
112#define VMDK_DDB_GEO_PCHS_HEADS "ddb.geometry.heads"
113
114/** VMDK descriptor DDB entry for PCHS sectors. */
115#define VMDK_DDB_GEO_PCHS_SECTORS "ddb.geometry.sectors"
116
117/** VMDK descriptor DDB entry for LCHS cylinders. */
118#define VMDK_DDB_GEO_LCHS_CYLINDERS "ddb.geometry.biosCylinders"
119
120/** VMDK descriptor DDB entry for LCHS heads. */
121#define VMDK_DDB_GEO_LCHS_HEADS "ddb.geometry.biosHeads"
122
123/** VMDK descriptor DDB entry for LCHS sectors. */
124#define VMDK_DDB_GEO_LCHS_SECTORS "ddb.geometry.biosSectors"
125
126/** VMDK descriptor DDB entry for image UUID. */
127#define VMDK_DDB_IMAGE_UUID "ddb.uuid.image"
128
129/** VMDK descriptor DDB entry for image modification UUID. */
130#define VMDK_DDB_MODIFICATION_UUID "ddb.uuid.modification"
131
132/** VMDK descriptor DDB entry for parent image UUID. */
133#define VMDK_DDB_PARENT_UUID "ddb.uuid.parent"
134
135/** VMDK descriptor DDB entry for parent image modification UUID. */
136#define VMDK_DDB_PARENT_MODIFICATION_UUID "ddb.uuid.parentmodification"
137
138/** No compression for streamOptimized files. */
139#define VMDK_COMPRESSION_NONE 0
140
141/** Deflate compression for streamOptimized files. */
142#define VMDK_COMPRESSION_DEFLATE 1
143
144/** Marker that the actual GD value is stored in the footer. */
145#define VMDK_GD_AT_END 0xffffffffffffffffULL
146
147/** Marker for end-of-stream in streamOptimized images. */
148#define VMDK_MARKER_EOS 0
149
150/** Marker for grain table block in streamOptimized images. */
151#define VMDK_MARKER_GT 1
152
153/** Marker for grain directory block in streamOptimized images. */
154#define VMDK_MARKER_GD 2
155
156/** Marker for footer in streamOptimized images. */
157#define VMDK_MARKER_FOOTER 3
158
159/** Marker for unknown purpose in streamOptimized images.
160 * Shows up in very recent images created by vSphere, but only sporadically.
161 * They "forgot" to document that one in the VMDK specification. */
162#define VMDK_MARKER_UNSPECIFIED 4
163
164/** Dummy marker for "don't check the marker value". */
165#define VMDK_MARKER_IGNORE 0xffffffffU
166
167/**
168 * Magic number for hosted images created by VMware Workstation 4, VMware
169 * Workstation 5, VMware Server or VMware Player. Not necessarily sparse.
170 */
171#define VMDK_SPARSE_MAGICNUMBER 0x564d444b /* 'V' 'M' 'D' 'K' */
172
173/** VMDK sector size in bytes. */
174#define VMDK_SECTOR_SIZE 512
175/** Max string buffer size for uint64_t with null term */
176#define UINT64_MAX_BUFF_SIZE 21
177/** Grain directory entry size in bytes */
178#define VMDK_GRAIN_DIR_ENTRY_SIZE 4
179/** Grain table size in bytes */
180#define VMDK_GRAIN_TABLE_SIZE 2048
181
182/**
183 * VMDK hosted binary extent header. The "Sparse" is a total misnomer, as
184 * this header is also used for monolithic flat images.
185 */
186#pragma pack(1)
187typedef struct SparseExtentHeader
188{
189 uint32_t magicNumber;
190 uint32_t version;
191 uint32_t flags;
192 uint64_t capacity;
193 uint64_t grainSize;
194 uint64_t descriptorOffset;
195 uint64_t descriptorSize;
196 uint32_t numGTEsPerGT;
197 uint64_t rgdOffset;
198 uint64_t gdOffset;
199 uint64_t overHead;
200 bool uncleanShutdown;
201 char singleEndLineChar;
202 char nonEndLineChar;
203 char doubleEndLineChar1;
204 char doubleEndLineChar2;
205 uint16_t compressAlgorithm;
206 uint8_t pad[433];
207} SparseExtentHeader;
208#pragma pack()
209
210/** The maximum allowed descriptor size in the extent header in sectors. */
211#define VMDK_SPARSE_DESCRIPTOR_SIZE_MAX UINT64_C(20480) /* 10MB */
212
213/** VMDK capacity for a single chunk when 2G splitting is turned on. Should be
214 * divisible by the default grain size (64K) */
215#define VMDK_2G_SPLIT_SIZE (2047 * 1024 * 1024)
216
217/** VMDK streamOptimized file format marker. The type field may or may not
218 * be actually valid, but there's always data to read there. */
219#pragma pack(1)
220typedef struct VMDKMARKER
221{
222 uint64_t uSector;
223 uint32_t cbSize;
224 uint32_t uType;
225} VMDKMARKER, *PVMDKMARKER;
226#pragma pack()
227
228
229/** Convert sector number/size to byte offset/size. */
230#define VMDK_SECTOR2BYTE(u) ((uint64_t)(u) << 9)
231
232/** Convert byte offset/size to sector number/size. */
233#define VMDK_BYTE2SECTOR(u) ((u) >> 9)
234
235/**
236 * VMDK extent type.
237 */
238typedef enum VMDKETYPE
239{
240 /** Hosted sparse extent. */
241 VMDKETYPE_HOSTED_SPARSE = 1,
242 /** Flat extent. */
243 VMDKETYPE_FLAT,
244 /** Zero extent. */
245 VMDKETYPE_ZERO,
246 /** VMFS extent, used by ESX. */
247 VMDKETYPE_VMFS
248} VMDKETYPE, *PVMDKETYPE;
249
250/**
251 * VMDK access type for a extent.
252 */
253typedef enum VMDKACCESS
254{
255 /** No access allowed. */
256 VMDKACCESS_NOACCESS = 0,
257 /** Read-only access. */
258 VMDKACCESS_READONLY,
259 /** Read-write access. */
260 VMDKACCESS_READWRITE
261} VMDKACCESS, *PVMDKACCESS;
262
263/** Forward declaration for PVMDKIMAGE. */
264typedef struct VMDKIMAGE *PVMDKIMAGE;
265
266/**
267 * Extents files entry. Used for opening a particular file only once.
268 */
269typedef struct VMDKFILE
270{
271 /** Pointer to file path. Local copy. */
272 const char *pszFilename;
273 /** Pointer to base name. Local copy. */
274 const char *pszBasename;
275 /** File open flags for consistency checking. */
276 unsigned fOpen;
277 /** Handle for sync/async file abstraction.*/
278 PVDIOSTORAGE pStorage;
279 /** Reference counter. */
280 unsigned uReferences;
281 /** Flag whether the file should be deleted on last close. */
282 bool fDelete;
283 /** Pointer to the image we belong to (for debugging purposes). */
284 PVMDKIMAGE pImage;
285 /** Pointer to next file descriptor. */
286 struct VMDKFILE *pNext;
287 /** Pointer to the previous file descriptor. */
288 struct VMDKFILE *pPrev;
289} VMDKFILE, *PVMDKFILE;
290
291/**
292 * VMDK extent data structure.
293 */
294typedef struct VMDKEXTENT
295{
296 /** File handle. */
297 PVMDKFILE pFile;
298 /** Base name of the image extent. */
299 const char *pszBasename;
300 /** Full name of the image extent. */
301 const char *pszFullname;
302 /** Number of sectors in this extent. */
303 uint64_t cSectors;
304 /** Number of sectors per block (grain in VMDK speak). */
305 uint64_t cSectorsPerGrain;
306 /** Starting sector number of descriptor. */
307 uint64_t uDescriptorSector;
308 /** Size of descriptor in sectors. */
309 uint64_t cDescriptorSectors;
310 /** Starting sector number of grain directory. */
311 uint64_t uSectorGD;
312 /** Starting sector number of redundant grain directory. */
313 uint64_t uSectorRGD;
314 /** Total number of metadata sectors. */
315 uint64_t cOverheadSectors;
316 /** Nominal size (i.e. as described by the descriptor) of this extent. */
317 uint64_t cNominalSectors;
318 /** Sector offset (i.e. as described by the descriptor) of this extent. */
319 uint64_t uSectorOffset;
320 /** Number of entries in a grain table. */
321 uint32_t cGTEntries;
322 /** Number of sectors reachable via a grain directory entry. */
323 uint32_t cSectorsPerGDE;
324 /** Number of entries in the grain directory. */
325 uint32_t cGDEntries;
326 /** Pointer to the next free sector. Legacy information. Do not use. */
327 uint32_t uFreeSector;
328 /** Number of this extent in the list of images. */
329 uint32_t uExtent;
330 /** Pointer to the descriptor (NULL if no descriptor in this extent). */
331 char *pDescData;
332 /** Pointer to the grain directory. */
333 uint32_t *pGD;
334 /** Pointer to the redundant grain directory. */
335 uint32_t *pRGD;
336 /** VMDK version of this extent. 1=1.0/1.1 */
337 uint32_t uVersion;
338 /** Type of this extent. */
339 VMDKETYPE enmType;
340 /** Access to this extent. */
341 VMDKACCESS enmAccess;
342 /** Flag whether this extent is marked as unclean. */
343 bool fUncleanShutdown;
344 /** Flag whether the metadata in the extent header needs to be updated. */
345 bool fMetaDirty;
346 /** Flag whether there is a footer in this extent. */
347 bool fFooter;
348 /** Compression type for this extent. */
349 uint16_t uCompression;
350 /** Append position for writing new grain. Only for sparse extents. */
351 uint64_t uAppendPosition;
352 /** Last grain which was accessed. Only for streamOptimized extents. */
353 uint32_t uLastGrainAccess;
354 /** Starting sector corresponding to the grain buffer. */
355 uint32_t uGrainSectorAbs;
356 /** Grain number corresponding to the grain buffer. */
357 uint32_t uGrain;
358 /** Actual size of the compressed data, only valid for reading. */
359 uint32_t cbGrainStreamRead;
360 /** Size of compressed grain buffer for streamOptimized extents. */
361 size_t cbCompGrain;
362 /** Compressed grain buffer for streamOptimized extents, with marker. */
363 void *pvCompGrain;
364 /** Decompressed grain buffer for streamOptimized extents. */
365 void *pvGrain;
366 /** Reference to the image in which this extent is used. Do not use this
367 * on a regular basis to avoid passing pImage references to functions
368 * explicitly. */
369 struct VMDKIMAGE *pImage;
370} VMDKEXTENT, *PVMDKEXTENT;
371
372/**
373 * Grain table cache size. Allocated per image.
374 */
375#define VMDK_GT_CACHE_SIZE 256
376
377/**
378 * Grain table block size. Smaller than an actual grain table block to allow
379 * more grain table blocks to be cached without having to allocate excessive
380 * amounts of memory for the cache.
381 */
382#define VMDK_GT_CACHELINE_SIZE 128
383
384
385/**
386 * Maximum number of lines in a descriptor file. Not worth the effort of
387 * making it variable. Descriptor files are generally very short (~20 lines),
388 * with the exception of sparse files split in 2G chunks, which need for the
389 * maximum size (almost 2T) exactly 1025 lines for the disk database.
390 */
391#define VMDK_DESCRIPTOR_LINES_MAX 1100U
392
393/**
394 * Parsed descriptor information. Allows easy access and update of the
395 * descriptor (whether separate file or not). Free form text files suck.
396 */
397typedef struct VMDKDESCRIPTOR
398{
399 /** Line number of first entry of the disk descriptor. */
400 unsigned uFirstDesc;
401 /** Line number of first entry in the extent description. */
402 unsigned uFirstExtent;
403 /** Line number of first disk database entry. */
404 unsigned uFirstDDB;
405 /** Total number of lines. */
406 unsigned cLines;
407 /** Total amount of memory available for the descriptor. */
408 size_t cbDescAlloc;
409 /** Set if descriptor has been changed and not yet written to disk. */
410 bool fDirty;
411 /** Array of pointers to the data in the descriptor. */
412 char *aLines[VMDK_DESCRIPTOR_LINES_MAX];
413 /** Array of line indices pointing to the next non-comment line. */
414 unsigned aNextLines[VMDK_DESCRIPTOR_LINES_MAX];
415} VMDKDESCRIPTOR, *PVMDKDESCRIPTOR;
416
417
418/**
419 * Cache entry for translating extent/sector to a sector number in that
420 * extent.
421 */
422typedef struct VMDKGTCACHEENTRY
423{
424 /** Extent number for which this entry is valid. */
425 uint32_t uExtent;
426 /** GT data block number. */
427 uint64_t uGTBlock;
428 /** Data part of the cache entry. */
429 uint32_t aGTData[VMDK_GT_CACHELINE_SIZE];
430} VMDKGTCACHEENTRY, *PVMDKGTCACHEENTRY;
431
432/**
433 * Cache data structure for blocks of grain table entries. For now this is a
434 * fixed size direct mapping cache, but this should be adapted to the size of
435 * the sparse image and maybe converted to a set-associative cache. The
436 * implementation below implements a write-through cache with write allocate.
437 */
438typedef struct VMDKGTCACHE
439{
440 /** Cache entries. */
441 VMDKGTCACHEENTRY aGTCache[VMDK_GT_CACHE_SIZE];
442 /** Number of cache entries (currently unused). */
443 unsigned cEntries;
444} VMDKGTCACHE, *PVMDKGTCACHE;
445
446/**
447 * Complete VMDK image data structure. Mainly a collection of extents and a few
448 * extra global data fields.
449 */
450typedef struct VMDKIMAGE
451{
452 /** Image name. */
453 const char *pszFilename;
454 /** Descriptor file if applicable. */
455 PVMDKFILE pFile;
456
457 /** Pointer to the per-disk VD interface list. */
458 PVDINTERFACE pVDIfsDisk;
459 /** Pointer to the per-image VD interface list. */
460 PVDINTERFACE pVDIfsImage;
461
462 /** Error interface. */
463 PVDINTERFACEERROR pIfError;
464 /** I/O interface. */
465 PVDINTERFACEIOINT pIfIo;
466
467
468 /** Pointer to the image extents. */
469 PVMDKEXTENT pExtents;
470 /** Number of image extents. */
471 unsigned cExtents;
472 /** Pointer to the files list, for opening a file referenced multiple
473 * times only once (happens mainly with raw partition access). */
474 PVMDKFILE pFiles;
475
476 /**
477 * Pointer to an array of segment entries for async I/O.
478 * This is an optimization because the task number to submit is not known
479 * and allocating/freeing an array in the read/write functions every time
480 * is too expensive.
481 */
482 PPDMDATASEG paSegments;
483 /** Entries available in the segments array. */
484 unsigned cSegments;
485
486 /** Open flags passed by VBoxHD layer. */
487 unsigned uOpenFlags;
488 /** Image flags defined during creation or determined during open. */
489 unsigned uImageFlags;
490 /** Total size of the image. */
491 uint64_t cbSize;
492 /** Physical geometry of this image. */
493 VDGEOMETRY PCHSGeometry;
494 /** Logical geometry of this image. */
495 VDGEOMETRY LCHSGeometry;
496 /** Image UUID. */
497 RTUUID ImageUuid;
498 /** Image modification UUID. */
499 RTUUID ModificationUuid;
500 /** Parent image UUID. */
501 RTUUID ParentUuid;
502 /** Parent image modification UUID. */
503 RTUUID ParentModificationUuid;
504
505 /** Pointer to grain table cache, if this image contains sparse extents. */
506 PVMDKGTCACHE pGTCache;
507 /** Pointer to the descriptor (NULL if no separate descriptor file). */
508 char *pDescData;
509 /** Allocation size of the descriptor file. */
510 size_t cbDescAlloc;
511 /** Parsed descriptor file content. */
512 VMDKDESCRIPTOR Descriptor;
513 /** The static region list. */
514 VDREGIONLIST RegionList;
515} VMDKIMAGE;
516
517
518/** State for the input/output callout of the inflate reader/deflate writer. */
519typedef struct VMDKCOMPRESSIO
520{
521 /* Image this operation relates to. */
522 PVMDKIMAGE pImage;
523 /* Current read position. */
524 ssize_t iOffset;
525 /* Size of the compressed grain buffer (available data). */
526 size_t cbCompGrain;
527 /* Pointer to the compressed grain buffer. */
528 void *pvCompGrain;
529} VMDKCOMPRESSIO;
530
531
532/** Tracks async grain allocation. */
533typedef struct VMDKGRAINALLOCASYNC
534{
535 /** Flag whether the allocation failed. */
536 bool fIoErr;
537 /** Current number of transfers pending.
538 * If reached 0 and there is an error the old state is restored. */
539 unsigned cIoXfersPending;
540 /** Sector number */
541 uint64_t uSector;
542 /** Flag whether the grain table needs to be updated. */
543 bool fGTUpdateNeeded;
544 /** Extent the allocation happens. */
545 PVMDKEXTENT pExtent;
546 /** Position of the new grain, required for the grain table update. */
547 uint64_t uGrainOffset;
548 /** Grain table sector. */
549 uint64_t uGTSector;
550 /** Backup grain table sector. */
551 uint64_t uRGTSector;
552} VMDKGRAINALLOCASYNC, *PVMDKGRAINALLOCASYNC;
553
554/**
555 * State information for vmdkRename() and helpers.
556 */
557typedef struct VMDKRENAMESTATE
558{
559 /** Array of old filenames. */
560 char **apszOldName;
561 /** Array of new filenames. */
562 char **apszNewName;
563 /** Array of new lines in the extent descriptor. */
564 char **apszNewLines;
565 /** Name of the old descriptor file if not a sparse image. */
566 char *pszOldDescName;
567 /** Flag whether we called vmdkFreeImage(). */
568 bool fImageFreed;
569 /** Flag whther the descriptor is embedded in the image (sparse) or
570 * in a separate file. */
571 bool fEmbeddedDesc;
572 /** Number of extents in the image. */
573 unsigned cExtents;
574 /** New base filename. */
575 char *pszNewBaseName;
576 /** The old base filename. */
577 char *pszOldBaseName;
578 /** New full filename. */
579 char *pszNewFullName;
580 /** Old full filename. */
581 char *pszOldFullName;
582 /** The old image name. */
583 const char *pszOldImageName;
584 /** Copy of the original VMDK descriptor. */
585 VMDKDESCRIPTOR DescriptorCopy;
586 /** Copy of the extent state for sparse images. */
587 VMDKEXTENT ExtentCopy;
588} VMDKRENAMESTATE;
589/** Pointer to a VMDK rename state. */
590typedef VMDKRENAMESTATE *PVMDKRENAMESTATE;
591
592
593/*********************************************************************************************************************************
594* Static Variables *
595*********************************************************************************************************************************/
596
597/** NULL-terminated array of supported file extensions. */
598static const VDFILEEXTENSION s_aVmdkFileExtensions[] =
599{
600 {"vmdk", VDTYPE_HDD},
601 {NULL, VDTYPE_INVALID}
602};
603
604/** NULL-terminated array of configuration option. */
605static const VDCONFIGINFO s_aVmdkConfigInfo[] =
606{
607 /* Options for VMDK raw disks */
608 { "RawDrive", NULL, VDCFGVALUETYPE_STRING, 0 },
609 { "Partitions", NULL, VDCFGVALUETYPE_STRING, 0 },
610 { "BootSector", NULL, VDCFGVALUETYPE_BYTES, 0 },
611 { "Relative", NULL, VDCFGVALUETYPE_INTEGER, 0 },
612
613 /* End of options list */
614 { NULL, NULL, VDCFGVALUETYPE_INTEGER, 0 }
615};
616
617
618/*********************************************************************************************************************************
619* Internal Functions *
620*********************************************************************************************************************************/
621
622static void vmdkFreeStreamBuffers(PVMDKEXTENT pExtent);
623static int vmdkFreeExtentData(PVMDKIMAGE pImage, PVMDKEXTENT pExtent,
624 bool fDelete);
625
626static int vmdkCreateExtents(PVMDKIMAGE pImage, unsigned cExtents);
627static int vmdkFlushImage(PVMDKIMAGE pImage, PVDIOCTX pIoCtx);
628static int vmdkSetImageComment(PVMDKIMAGE pImage, const char *pszComment);
629static int vmdkFreeImage(PVMDKIMAGE pImage, bool fDelete, bool fFlush);
630
631static DECLCALLBACK(int) vmdkAllocGrainComplete(void *pBackendData, PVDIOCTX pIoCtx,
632 void *pvUser, int rcReq);
633
634/**
635 * Internal: open a file (using a file descriptor cache to ensure each file
636 * is only opened once - anything else can cause locking problems).
637 */
638static int vmdkFileOpen(PVMDKIMAGE pImage, PVMDKFILE *ppVmdkFile,
639 const char *pszBasename, const char *pszFilename, uint32_t fOpen)
640{
641 int rc = VINF_SUCCESS;
642 PVMDKFILE pVmdkFile;
643
644 for (pVmdkFile = pImage->pFiles;
645 pVmdkFile != NULL;
646 pVmdkFile = pVmdkFile->pNext)
647 {
648 if (!strcmp(pszFilename, pVmdkFile->pszFilename))
649 {
650 Assert(fOpen == pVmdkFile->fOpen);
651 pVmdkFile->uReferences++;
652
653 *ppVmdkFile = pVmdkFile;
654
655 return rc;
656 }
657 }
658
659 /* If we get here, there's no matching entry in the cache. */
660 pVmdkFile = (PVMDKFILE)RTMemAllocZ(sizeof(VMDKFILE));
661 if (!pVmdkFile)
662 {
663 *ppVmdkFile = NULL;
664 return VERR_NO_MEMORY;
665 }
666
667 pVmdkFile->pszFilename = RTStrDup(pszFilename);
668 if (!pVmdkFile->pszFilename)
669 {
670 RTMemFree(pVmdkFile);
671 *ppVmdkFile = NULL;
672 return VERR_NO_MEMORY;
673 }
674
675 if (pszBasename)
676 {
677 pVmdkFile->pszBasename = RTStrDup(pszBasename);
678 if (!pVmdkFile->pszBasename)
679 {
680 RTStrFree((char *)(void *)pVmdkFile->pszFilename);
681 RTMemFree(pVmdkFile);
682 *ppVmdkFile = NULL;
683 return VERR_NO_MEMORY;
684 }
685 }
686
687 pVmdkFile->fOpen = fOpen;
688
689 rc = vdIfIoIntFileOpen(pImage->pIfIo, pszFilename, fOpen,
690 &pVmdkFile->pStorage);
691 if (RT_SUCCESS(rc))
692 {
693 pVmdkFile->uReferences = 1;
694 pVmdkFile->pImage = pImage;
695 pVmdkFile->pNext = pImage->pFiles;
696 if (pImage->pFiles)
697 pImage->pFiles->pPrev = pVmdkFile;
698 pImage->pFiles = pVmdkFile;
699 *ppVmdkFile = pVmdkFile;
700 }
701 else
702 {
703 RTStrFree((char *)(void *)pVmdkFile->pszFilename);
704 RTMemFree(pVmdkFile);
705 *ppVmdkFile = NULL;
706 }
707
708 return rc;
709}
710
711/**
712 * Internal: close a file, updating the file descriptor cache.
713 */
714static int vmdkFileClose(PVMDKIMAGE pImage, PVMDKFILE *ppVmdkFile, bool fDelete)
715{
716 int rc = VINF_SUCCESS;
717 PVMDKFILE pVmdkFile = *ppVmdkFile;
718
719 AssertPtr(pVmdkFile);
720
721 pVmdkFile->fDelete |= fDelete;
722 Assert(pVmdkFile->uReferences);
723 pVmdkFile->uReferences--;
724 if (pVmdkFile->uReferences == 0)
725 {
726 PVMDKFILE pPrev;
727 PVMDKFILE pNext;
728
729 /* Unchain the element from the list. */
730 pPrev = pVmdkFile->pPrev;
731 pNext = pVmdkFile->pNext;
732
733 if (pNext)
734 pNext->pPrev = pPrev;
735 if (pPrev)
736 pPrev->pNext = pNext;
737 else
738 pImage->pFiles = pNext;
739
740 rc = vdIfIoIntFileClose(pImage->pIfIo, pVmdkFile->pStorage);
741
742 bool fFileDel = pVmdkFile->fDelete;
743 if ( pVmdkFile->pszBasename
744 && fFileDel)
745 {
746 const char *pszSuffix = RTPathSuffix(pVmdkFile->pszBasename);
747 if ( RTPathHasPath(pVmdkFile->pszBasename)
748 || !pszSuffix
749 || ( strcmp(pszSuffix, ".vmdk")
750 && strcmp(pszSuffix, ".bin")
751 && strcmp(pszSuffix, ".img")))
752 fFileDel = false;
753 }
754
755 if (fFileDel)
756 {
757 int rc2 = vdIfIoIntFileDelete(pImage->pIfIo, pVmdkFile->pszFilename);
758 if (RT_SUCCESS(rc))
759 rc = rc2;
760 }
761 else if (pVmdkFile->fDelete)
762 LogRel(("VMDK: Denying deletion of %s\n", pVmdkFile->pszBasename));
763 RTStrFree((char *)(void *)pVmdkFile->pszFilename);
764 if (pVmdkFile->pszBasename)
765 RTStrFree((char *)(void *)pVmdkFile->pszBasename);
766 RTMemFree(pVmdkFile);
767 }
768
769 *ppVmdkFile = NULL;
770 return rc;
771}
772
773/*#define VMDK_USE_BLOCK_DECOMP_API - test and enable */
774#ifndef VMDK_USE_BLOCK_DECOMP_API
775static DECLCALLBACK(int) vmdkFileInflateHelper(void *pvUser, void *pvBuf, size_t cbBuf, size_t *pcbBuf)
776{
777 VMDKCOMPRESSIO *pInflateState = (VMDKCOMPRESSIO *)pvUser;
778 size_t cbInjected = 0;
779
780 Assert(cbBuf);
781 if (pInflateState->iOffset < 0)
782 {
783 *(uint8_t *)pvBuf = RTZIPTYPE_ZLIB;
784 pvBuf = (uint8_t *)pvBuf + 1;
785 cbBuf--;
786 cbInjected = 1;
787 pInflateState->iOffset = RT_UOFFSETOF(VMDKMARKER, uType);
788 }
789 if (!cbBuf)
790 {
791 if (pcbBuf)
792 *pcbBuf = cbInjected;
793 return VINF_SUCCESS;
794 }
795 cbBuf = RT_MIN(cbBuf, pInflateState->cbCompGrain - pInflateState->iOffset);
796 memcpy(pvBuf,
797 (uint8_t *)pInflateState->pvCompGrain + pInflateState->iOffset,
798 cbBuf);
799 pInflateState->iOffset += cbBuf;
800 Assert(pcbBuf);
801 *pcbBuf = cbBuf + cbInjected;
802 return VINF_SUCCESS;
803}
804#endif
805
806/**
807 * Internal: read from a file and inflate the compressed data,
808 * distinguishing between async and normal operation
809 */
810DECLINLINE(int) vmdkFileInflateSync(PVMDKIMAGE pImage, PVMDKEXTENT pExtent,
811 uint64_t uOffset, void *pvBuf,
812 size_t cbToRead, const void *pcvMarker,
813 uint64_t *puLBA, uint32_t *pcbMarkerData)
814{
815 int rc;
816#ifndef VMDK_USE_BLOCK_DECOMP_API
817 PRTZIPDECOMP pZip = NULL;
818#endif
819 VMDKMARKER *pMarker = (VMDKMARKER *)pExtent->pvCompGrain;
820 size_t cbCompSize, cbActuallyRead;
821
822 if (!pcvMarker)
823 {
824 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage,
825 uOffset, pMarker, RT_UOFFSETOF(VMDKMARKER, uType));
826 if (RT_FAILURE(rc))
827 return rc;
828 }
829 else
830 {
831 memcpy(pMarker, pcvMarker, RT_UOFFSETOF(VMDKMARKER, uType));
832 /* pcvMarker endianness has already been partially transformed, fix it */
833 pMarker->uSector = RT_H2LE_U64(pMarker->uSector);
834 pMarker->cbSize = RT_H2LE_U32(pMarker->cbSize);
835 }
836
837 cbCompSize = RT_LE2H_U32(pMarker->cbSize);
838 if (cbCompSize == 0)
839 {
840 AssertMsgFailed(("VMDK: corrupted marker\n"));
841 return VERR_VD_VMDK_INVALID_FORMAT;
842 }
843
844 /* Sanity check - the expansion ratio should be much less than 2. */
845 Assert(cbCompSize < 2 * cbToRead);
846 if (cbCompSize >= 2 * cbToRead)
847 return VERR_VD_VMDK_INVALID_FORMAT;
848
849 /* Compressed grain marker. Data follows immediately. */
850 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage,
851 uOffset + RT_UOFFSETOF(VMDKMARKER, uType),
852 (uint8_t *)pExtent->pvCompGrain
853 + RT_UOFFSETOF(VMDKMARKER, uType),
854 RT_ALIGN_Z( cbCompSize
855 + RT_UOFFSETOF(VMDKMARKER, uType),
856 512)
857 - RT_UOFFSETOF(VMDKMARKER, uType));
858
859 if (puLBA)
860 *puLBA = RT_LE2H_U64(pMarker->uSector);
861 if (pcbMarkerData)
862 *pcbMarkerData = RT_ALIGN( cbCompSize
863 + RT_UOFFSETOF(VMDKMARKER, uType),
864 512);
865
866#ifdef VMDK_USE_BLOCK_DECOMP_API
867 rc = RTZipBlockDecompress(RTZIPTYPE_ZLIB, 0 /*fFlags*/,
868 pExtent->pvCompGrain, cbCompSize + RT_UOFFSETOF(VMDKMARKER, uType), NULL,
869 pvBuf, cbToRead, &cbActuallyRead);
870#else
871 VMDKCOMPRESSIO InflateState;
872 InflateState.pImage = pImage;
873 InflateState.iOffset = -1;
874 InflateState.cbCompGrain = cbCompSize + RT_UOFFSETOF(VMDKMARKER, uType);
875 InflateState.pvCompGrain = pExtent->pvCompGrain;
876
877 rc = RTZipDecompCreate(&pZip, &InflateState, vmdkFileInflateHelper);
878 if (RT_FAILURE(rc))
879 return rc;
880 rc = RTZipDecompress(pZip, pvBuf, cbToRead, &cbActuallyRead);
881 RTZipDecompDestroy(pZip);
882#endif /* !VMDK_USE_BLOCK_DECOMP_API */
883 if (RT_FAILURE(rc))
884 {
885 if (rc == VERR_ZIP_CORRUPTED)
886 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: Compressed image is corrupted '%s'"), pExtent->pszFullname);
887 return rc;
888 }
889 if (cbActuallyRead != cbToRead)
890 rc = VERR_VD_VMDK_INVALID_FORMAT;
891 return rc;
892}
893
894static DECLCALLBACK(int) vmdkFileDeflateHelper(void *pvUser, const void *pvBuf, size_t cbBuf)
895{
896 VMDKCOMPRESSIO *pDeflateState = (VMDKCOMPRESSIO *)pvUser;
897
898 Assert(cbBuf);
899 if (pDeflateState->iOffset < 0)
900 {
901 pvBuf = (const uint8_t *)pvBuf + 1;
902 cbBuf--;
903 pDeflateState->iOffset = RT_UOFFSETOF(VMDKMARKER, uType);
904 }
905 if (!cbBuf)
906 return VINF_SUCCESS;
907 if (pDeflateState->iOffset + cbBuf > pDeflateState->cbCompGrain)
908 return VERR_BUFFER_OVERFLOW;
909 memcpy((uint8_t *)pDeflateState->pvCompGrain + pDeflateState->iOffset,
910 pvBuf, cbBuf);
911 pDeflateState->iOffset += cbBuf;
912 return VINF_SUCCESS;
913}
914
915/**
916 * Internal: deflate the uncompressed data and write to a file,
917 * distinguishing between async and normal operation
918 */
919DECLINLINE(int) vmdkFileDeflateSync(PVMDKIMAGE pImage, PVMDKEXTENT pExtent,
920 uint64_t uOffset, const void *pvBuf,
921 size_t cbToWrite, uint64_t uLBA,
922 uint32_t *pcbMarkerData)
923{
924 int rc;
925 PRTZIPCOMP pZip = NULL;
926 VMDKCOMPRESSIO DeflateState;
927
928 DeflateState.pImage = pImage;
929 DeflateState.iOffset = -1;
930 DeflateState.cbCompGrain = pExtent->cbCompGrain;
931 DeflateState.pvCompGrain = pExtent->pvCompGrain;
932
933 rc = RTZipCompCreate(&pZip, &DeflateState, vmdkFileDeflateHelper,
934 RTZIPTYPE_ZLIB, RTZIPLEVEL_DEFAULT);
935 if (RT_FAILURE(rc))
936 return rc;
937 rc = RTZipCompress(pZip, pvBuf, cbToWrite);
938 if (RT_SUCCESS(rc))
939 rc = RTZipCompFinish(pZip);
940 RTZipCompDestroy(pZip);
941 if (RT_SUCCESS(rc))
942 {
943 Assert( DeflateState.iOffset > 0
944 && (size_t)DeflateState.iOffset <= DeflateState.cbCompGrain);
945
946 /* pad with zeroes to get to a full sector size */
947 uint32_t uSize = DeflateState.iOffset;
948 if (uSize % 512)
949 {
950 uint32_t uSizeAlign = RT_ALIGN(uSize, 512);
951 memset((uint8_t *)pExtent->pvCompGrain + uSize, '\0',
952 uSizeAlign - uSize);
953 uSize = uSizeAlign;
954 }
955
956 if (pcbMarkerData)
957 *pcbMarkerData = uSize;
958
959 /* Compressed grain marker. Data follows immediately. */
960 VMDKMARKER *pMarker = (VMDKMARKER *)pExtent->pvCompGrain;
961 pMarker->uSector = RT_H2LE_U64(uLBA);
962 pMarker->cbSize = RT_H2LE_U32( DeflateState.iOffset
963 - RT_UOFFSETOF(VMDKMARKER, uType));
964 rc = vdIfIoIntFileWriteSync(pImage->pIfIo, pExtent->pFile->pStorage,
965 uOffset, pMarker, uSize);
966 if (RT_FAILURE(rc))
967 return rc;
968 }
969 return rc;
970}
971
972
973/**
974 * Internal: check if all files are closed, prevent leaking resources.
975 */
976static int vmdkFileCheckAllClose(PVMDKIMAGE pImage)
977{
978 int rc = VINF_SUCCESS, rc2;
979 PVMDKFILE pVmdkFile;
980
981 Assert(pImage->pFiles == NULL);
982 for (pVmdkFile = pImage->pFiles;
983 pVmdkFile != NULL;
984 pVmdkFile = pVmdkFile->pNext)
985 {
986 LogRel(("VMDK: leaking reference to file \"%s\"\n",
987 pVmdkFile->pszFilename));
988 pImage->pFiles = pVmdkFile->pNext;
989
990 rc2 = vmdkFileClose(pImage, &pVmdkFile, pVmdkFile->fDelete);
991
992 if (RT_SUCCESS(rc))
993 rc = rc2;
994 }
995 return rc;
996}
997
998/**
999 * Internal: truncate a string (at a UTF8 code point boundary) and encode the
1000 * critical non-ASCII characters.
1001 */
1002static char *vmdkEncodeString(const char *psz)
1003{
1004 char szEnc[VMDK_ENCODED_COMMENT_MAX + 3];
1005 char *pszDst = szEnc;
1006
1007 AssertPtr(psz);
1008
1009 for (; *psz; psz = RTStrNextCp(psz))
1010 {
1011 char *pszDstPrev = pszDst;
1012 RTUNICP Cp = RTStrGetCp(psz);
1013 if (Cp == '\\')
1014 {
1015 pszDst = RTStrPutCp(pszDst, Cp);
1016 pszDst = RTStrPutCp(pszDst, Cp);
1017 }
1018 else if (Cp == '\n')
1019 {
1020 pszDst = RTStrPutCp(pszDst, '\\');
1021 pszDst = RTStrPutCp(pszDst, 'n');
1022 }
1023 else if (Cp == '\r')
1024 {
1025 pszDst = RTStrPutCp(pszDst, '\\');
1026 pszDst = RTStrPutCp(pszDst, 'r');
1027 }
1028 else
1029 pszDst = RTStrPutCp(pszDst, Cp);
1030 if (pszDst - szEnc >= VMDK_ENCODED_COMMENT_MAX - 1)
1031 {
1032 pszDst = pszDstPrev;
1033 break;
1034 }
1035 }
1036 *pszDst = '\0';
1037 return RTStrDup(szEnc);
1038}
1039
1040/**
1041 * Internal: decode a string and store it into the specified string.
1042 */
1043static int vmdkDecodeString(const char *pszEncoded, char *psz, size_t cb)
1044{
1045 int rc = VINF_SUCCESS;
1046 char szBuf[4];
1047
1048 if (!cb)
1049 return VERR_BUFFER_OVERFLOW;
1050
1051 AssertPtr(psz);
1052
1053 for (; *pszEncoded; pszEncoded = RTStrNextCp(pszEncoded))
1054 {
1055 char *pszDst = szBuf;
1056 RTUNICP Cp = RTStrGetCp(pszEncoded);
1057 if (Cp == '\\')
1058 {
1059 pszEncoded = RTStrNextCp(pszEncoded);
1060 RTUNICP CpQ = RTStrGetCp(pszEncoded);
1061 if (CpQ == 'n')
1062 RTStrPutCp(pszDst, '\n');
1063 else if (CpQ == 'r')
1064 RTStrPutCp(pszDst, '\r');
1065 else if (CpQ == '\0')
1066 {
1067 rc = VERR_VD_VMDK_INVALID_HEADER;
1068 break;
1069 }
1070 else
1071 RTStrPutCp(pszDst, CpQ);
1072 }
1073 else
1074 pszDst = RTStrPutCp(pszDst, Cp);
1075
1076 /* Need to leave space for terminating NUL. */
1077 if ((size_t)(pszDst - szBuf) + 1 >= cb)
1078 {
1079 rc = VERR_BUFFER_OVERFLOW;
1080 break;
1081 }
1082 memcpy(psz, szBuf, pszDst - szBuf);
1083 psz += pszDst - szBuf;
1084 }
1085 *psz = '\0';
1086 return rc;
1087}
1088
1089/**
1090 * Internal: free all buffers associated with grain directories.
1091 */
1092static void vmdkFreeGrainDirectory(PVMDKEXTENT pExtent)
1093{
1094 if (pExtent->pGD)
1095 {
1096 RTMemFree(pExtent->pGD);
1097 pExtent->pGD = NULL;
1098 }
1099 if (pExtent->pRGD)
1100 {
1101 RTMemFree(pExtent->pRGD);
1102 pExtent->pRGD = NULL;
1103 }
1104}
1105
1106/**
1107 * Internal: allocate the compressed/uncompressed buffers for streamOptimized
1108 * images.
1109 */
1110static int vmdkAllocStreamBuffers(PVMDKIMAGE pImage, PVMDKEXTENT pExtent)
1111{
1112 int rc = VINF_SUCCESS;
1113
1114 if (pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED)
1115 {
1116 /* streamOptimized extents need a compressed grain buffer, which must
1117 * be big enough to hold uncompressible data (which needs ~8 bytes
1118 * more than the uncompressed data), the marker and padding. */
1119 pExtent->cbCompGrain = RT_ALIGN_Z( VMDK_SECTOR2BYTE(pExtent->cSectorsPerGrain)
1120 + 8 + sizeof(VMDKMARKER), 512);
1121 pExtent->pvCompGrain = RTMemAlloc(pExtent->cbCompGrain);
1122 if (RT_LIKELY(pExtent->pvCompGrain))
1123 {
1124 /* streamOptimized extents need a decompressed grain buffer. */
1125 pExtent->pvGrain = RTMemAlloc(VMDK_SECTOR2BYTE(pExtent->cSectorsPerGrain));
1126 if (!pExtent->pvGrain)
1127 rc = VERR_NO_MEMORY;
1128 }
1129 else
1130 rc = VERR_NO_MEMORY;
1131 }
1132
1133 if (RT_FAILURE(rc))
1134 vmdkFreeStreamBuffers(pExtent);
1135 return rc;
1136}
1137
1138/**
1139 * Internal: allocate all buffers associated with grain directories.
1140 */
1141static int vmdkAllocGrainDirectory(PVMDKIMAGE pImage, PVMDKEXTENT pExtent)
1142{
1143 RT_NOREF1(pImage);
1144 int rc = VINF_SUCCESS;
1145 size_t cbGD = pExtent->cGDEntries * sizeof(uint32_t);
1146
1147 pExtent->pGD = (uint32_t *)RTMemAllocZ(cbGD);
1148 if (RT_LIKELY(pExtent->pGD))
1149 {
1150 if (pExtent->uSectorRGD)
1151 {
1152 pExtent->pRGD = (uint32_t *)RTMemAllocZ(cbGD);
1153 if (RT_UNLIKELY(!pExtent->pRGD))
1154 rc = VERR_NO_MEMORY;
1155 }
1156 }
1157 else
1158 rc = VERR_NO_MEMORY;
1159
1160 if (RT_FAILURE(rc))
1161 vmdkFreeGrainDirectory(pExtent);
1162 return rc;
1163}
1164
1165/**
1166 * Converts the grain directory from little to host endianess.
1167 *
1168 * @param pGD The grain directory.
1169 * @param cGDEntries Number of entries in the grain directory to convert.
1170 */
1171DECLINLINE(void) vmdkGrainDirectoryConvToHost(uint32_t *pGD, uint32_t cGDEntries)
1172{
1173 uint32_t *pGDTmp = pGD;
1174
1175 for (uint32_t i = 0; i < cGDEntries; i++, pGDTmp++)
1176 *pGDTmp = RT_LE2H_U32(*pGDTmp);
1177}
1178
1179/**
1180 * Read the grain directory and allocated grain tables verifying them against
1181 * their back up copies if available.
1182 *
1183 * @returns VBox status code.
1184 * @param pImage Image instance data.
1185 * @param pExtent The VMDK extent.
1186 */
1187static int vmdkReadGrainDirectory(PVMDKIMAGE pImage, PVMDKEXTENT pExtent)
1188{
1189 int rc = VINF_SUCCESS;
1190 size_t cbGD = pExtent->cGDEntries * sizeof(uint32_t);
1191
1192 AssertReturn(( pExtent->enmType == VMDKETYPE_HOSTED_SPARSE
1193 && pExtent->uSectorGD != VMDK_GD_AT_END
1194 && pExtent->uSectorRGD != VMDK_GD_AT_END), VERR_INTERNAL_ERROR);
1195
1196 rc = vmdkAllocGrainDirectory(pImage, pExtent);
1197 if (RT_SUCCESS(rc))
1198 {
1199 /* The VMDK 1.1 spec seems to talk about compressed grain directories,
1200 * but in reality they are not compressed. */
1201 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage,
1202 VMDK_SECTOR2BYTE(pExtent->uSectorGD),
1203 pExtent->pGD, cbGD);
1204 if (RT_SUCCESS(rc))
1205 {
1206 vmdkGrainDirectoryConvToHost(pExtent->pGD, pExtent->cGDEntries);
1207
1208 if ( pExtent->uSectorRGD
1209 && !(pImage->uOpenFlags & VD_OPEN_FLAGS_SKIP_CONSISTENCY_CHECKS))
1210 {
1211 /* The VMDK 1.1 spec seems to talk about compressed grain directories,
1212 * but in reality they are not compressed. */
1213 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage,
1214 VMDK_SECTOR2BYTE(pExtent->uSectorRGD),
1215 pExtent->pRGD, cbGD);
1216 if (RT_SUCCESS(rc))
1217 {
1218 vmdkGrainDirectoryConvToHost(pExtent->pRGD, pExtent->cGDEntries);
1219
1220 /* Check grain table and redundant grain table for consistency. */
1221 size_t cbGT = pExtent->cGTEntries * sizeof(uint32_t);
1222 size_t cbGTBuffers = cbGT; /* Start with space for one GT. */
1223 size_t cbGTBuffersMax = _1M;
1224
1225 uint32_t *pTmpGT1 = (uint32_t *)RTMemAlloc(cbGTBuffers);
1226 uint32_t *pTmpGT2 = (uint32_t *)RTMemAlloc(cbGTBuffers);
1227
1228 if ( !pTmpGT1
1229 || !pTmpGT2)
1230 rc = VERR_NO_MEMORY;
1231
1232 size_t i = 0;
1233 uint32_t *pGDTmp = pExtent->pGD;
1234 uint32_t *pRGDTmp = pExtent->pRGD;
1235
1236 /* Loop through all entries. */
1237 while (i < pExtent->cGDEntries)
1238 {
1239 uint32_t uGTStart = *pGDTmp;
1240 uint32_t uRGTStart = *pRGDTmp;
1241 size_t cbGTRead = cbGT;
1242
1243 /* If no grain table is allocated skip the entry. */
1244 if (*pGDTmp == 0 && *pRGDTmp == 0)
1245 {
1246 i++;
1247 continue;
1248 }
1249
1250 if (*pGDTmp == 0 || *pRGDTmp == 0 || *pGDTmp == *pRGDTmp)
1251 {
1252 /* Just one grain directory entry refers to a not yet allocated
1253 * grain table or both grain directory copies refer to the same
1254 * grain table. Not allowed. */
1255 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
1256 N_("VMDK: inconsistent references to grain directory in '%s'"), pExtent->pszFullname);
1257 break;
1258 }
1259
1260 i++;
1261 pGDTmp++;
1262 pRGDTmp++;
1263
1264 /*
1265 * Read a few tables at once if adjacent to decrease the number
1266 * of I/O requests. Read at maximum 1MB at once.
1267 */
1268 while ( i < pExtent->cGDEntries
1269 && cbGTRead < cbGTBuffersMax)
1270 {
1271 /* If no grain table is allocated skip the entry. */
1272 if (*pGDTmp == 0 && *pRGDTmp == 0)
1273 {
1274 i++;
1275 continue;
1276 }
1277
1278 if (*pGDTmp == 0 || *pRGDTmp == 0 || *pGDTmp == *pRGDTmp)
1279 {
1280 /* Just one grain directory entry refers to a not yet allocated
1281 * grain table or both grain directory copies refer to the same
1282 * grain table. Not allowed. */
1283 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
1284 N_("VMDK: inconsistent references to grain directory in '%s'"), pExtent->pszFullname);
1285 break;
1286 }
1287
1288 /* Check that the start offsets are adjacent.*/
1289 if ( VMDK_SECTOR2BYTE(uGTStart) + cbGTRead != VMDK_SECTOR2BYTE(*pGDTmp)
1290 || VMDK_SECTOR2BYTE(uRGTStart) + cbGTRead != VMDK_SECTOR2BYTE(*pRGDTmp))
1291 break;
1292
1293 i++;
1294 pGDTmp++;
1295 pRGDTmp++;
1296 cbGTRead += cbGT;
1297 }
1298
1299 /* Increase buffers if required. */
1300 if ( RT_SUCCESS(rc)
1301 && cbGTBuffers < cbGTRead)
1302 {
1303 uint32_t *pTmp;
1304 pTmp = (uint32_t *)RTMemRealloc(pTmpGT1, cbGTRead);
1305 if (pTmp)
1306 {
1307 pTmpGT1 = pTmp;
1308 pTmp = (uint32_t *)RTMemRealloc(pTmpGT2, cbGTRead);
1309 if (pTmp)
1310 pTmpGT2 = pTmp;
1311 else
1312 rc = VERR_NO_MEMORY;
1313 }
1314 else
1315 rc = VERR_NO_MEMORY;
1316
1317 if (rc == VERR_NO_MEMORY)
1318 {
1319 /* Reset to the old values. */
1320 rc = VINF_SUCCESS;
1321 i -= cbGTRead / cbGT;
1322 cbGTRead = cbGT;
1323
1324 /* Don't try to increase the buffer again in the next run. */
1325 cbGTBuffersMax = cbGTBuffers;
1326 }
1327 }
1328
1329 if (RT_SUCCESS(rc))
1330 {
1331 /* The VMDK 1.1 spec seems to talk about compressed grain tables,
1332 * but in reality they are not compressed. */
1333 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage,
1334 VMDK_SECTOR2BYTE(uGTStart),
1335 pTmpGT1, cbGTRead);
1336 if (RT_FAILURE(rc))
1337 {
1338 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
1339 N_("VMDK: error reading grain table in '%s'"), pExtent->pszFullname);
1340 break;
1341 }
1342 /* The VMDK 1.1 spec seems to talk about compressed grain tables,
1343 * but in reality they are not compressed. */
1344 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage,
1345 VMDK_SECTOR2BYTE(uRGTStart),
1346 pTmpGT2, cbGTRead);
1347 if (RT_FAILURE(rc))
1348 {
1349 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
1350 N_("VMDK: error reading backup grain table in '%s'"), pExtent->pszFullname);
1351 break;
1352 }
1353 if (memcmp(pTmpGT1, pTmpGT2, cbGTRead))
1354 {
1355 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
1356 N_("VMDK: inconsistency between grain table and backup grain table in '%s'"), pExtent->pszFullname);
1357 break;
1358 }
1359 }
1360 } /* while (i < pExtent->cGDEntries) */
1361
1362 /** @todo figure out what to do for unclean VMDKs. */
1363 if (pTmpGT1)
1364 RTMemFree(pTmpGT1);
1365 if (pTmpGT2)
1366 RTMemFree(pTmpGT2);
1367 }
1368 else
1369 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
1370 N_("VMDK: could not read redundant grain directory in '%s'"), pExtent->pszFullname);
1371 }
1372 }
1373 else
1374 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
1375 N_("VMDK: could not read grain directory in '%s': %Rrc"), pExtent->pszFullname, rc);
1376 }
1377
1378 if (RT_FAILURE(rc))
1379 vmdkFreeGrainDirectory(pExtent);
1380 return rc;
1381}
1382
1383/**
1384 * Creates a new grain directory for the given extent at the given start sector.
1385 *
1386 * @returns VBox status code.
1387 * @param pImage Image instance data.
1388 * @param pExtent The VMDK extent.
1389 * @param uStartSector Where the grain directory should be stored in the image.
1390 * @param fPreAlloc Flag whether to pre allocate the grain tables at this point.
1391 */
1392static int vmdkCreateGrainDirectory(PVMDKIMAGE pImage, PVMDKEXTENT pExtent,
1393 uint64_t uStartSector, bool fPreAlloc)
1394{
1395 int rc = VINF_SUCCESS;
1396 unsigned i;
1397 size_t cbGD = pExtent->cGDEntries * sizeof(uint32_t);
1398 size_t cbGDRounded = RT_ALIGN_64(cbGD, 512);
1399 size_t cbGTRounded;
1400 uint64_t cbOverhead;
1401
1402 if (fPreAlloc)
1403 {
1404 cbGTRounded = RT_ALIGN_64(pExtent->cGDEntries * pExtent->cGTEntries * sizeof(uint32_t), 512);
1405 cbOverhead = VMDK_SECTOR2BYTE(uStartSector) + cbGDRounded + cbGTRounded;
1406 }
1407 else
1408 {
1409 /* Use a dummy start sector for layout computation. */
1410 if (uStartSector == VMDK_GD_AT_END)
1411 uStartSector = 1;
1412 cbGTRounded = 0;
1413 cbOverhead = VMDK_SECTOR2BYTE(uStartSector) + cbGDRounded;
1414 }
1415
1416 /* For streamOptimized extents there is only one grain directory,
1417 * and for all others take redundant grain directory into account. */
1418 if (pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED)
1419 {
1420 cbOverhead = RT_ALIGN_64(cbOverhead,
1421 VMDK_SECTOR2BYTE(pExtent->cSectorsPerGrain));
1422 }
1423 else
1424 {
1425 cbOverhead += cbGDRounded + cbGTRounded;
1426 cbOverhead = RT_ALIGN_64(cbOverhead,
1427 VMDK_SECTOR2BYTE(pExtent->cSectorsPerGrain));
1428 rc = vdIfIoIntFileSetSize(pImage->pIfIo, pExtent->pFile->pStorage, cbOverhead);
1429 }
1430
1431 if (RT_SUCCESS(rc))
1432 {
1433 pExtent->uAppendPosition = cbOverhead;
1434 pExtent->cOverheadSectors = VMDK_BYTE2SECTOR(cbOverhead);
1435
1436 if (pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED)
1437 {
1438 pExtent->uSectorRGD = 0;
1439 pExtent->uSectorGD = uStartSector;
1440 }
1441 else
1442 {
1443 pExtent->uSectorRGD = uStartSector;
1444 pExtent->uSectorGD = uStartSector + VMDK_BYTE2SECTOR(cbGDRounded + cbGTRounded);
1445 }
1446
1447 rc = vmdkAllocStreamBuffers(pImage, pExtent);
1448 if (RT_SUCCESS(rc))
1449 {
1450 rc = vmdkAllocGrainDirectory(pImage, pExtent);
1451 if ( RT_SUCCESS(rc)
1452 && fPreAlloc)
1453 {
1454 uint32_t uGTSectorLE;
1455 uint64_t uOffsetSectors;
1456
1457 if (pExtent->pRGD)
1458 {
1459 uOffsetSectors = pExtent->uSectorRGD + VMDK_BYTE2SECTOR(cbGDRounded);
1460 for (i = 0; i < pExtent->cGDEntries; i++)
1461 {
1462 pExtent->pRGD[i] = uOffsetSectors;
1463 uGTSectorLE = RT_H2LE_U64(uOffsetSectors);
1464 /* Write the redundant grain directory entry to disk. */
1465 rc = vdIfIoIntFileWriteSync(pImage->pIfIo, pExtent->pFile->pStorage,
1466 VMDK_SECTOR2BYTE(pExtent->uSectorRGD) + i * sizeof(uGTSectorLE),
1467 &uGTSectorLE, sizeof(uGTSectorLE));
1468 if (RT_FAILURE(rc))
1469 {
1470 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: cannot write new redundant grain directory entry in '%s'"), pExtent->pszFullname);
1471 break;
1472 }
1473 uOffsetSectors += VMDK_BYTE2SECTOR(pExtent->cGTEntries * sizeof(uint32_t));
1474 }
1475 }
1476
1477 if (RT_SUCCESS(rc))
1478 {
1479 uOffsetSectors = pExtent->uSectorGD + VMDK_BYTE2SECTOR(cbGDRounded);
1480 for (i = 0; i < pExtent->cGDEntries; i++)
1481 {
1482 pExtent->pGD[i] = uOffsetSectors;
1483 uGTSectorLE = RT_H2LE_U64(uOffsetSectors);
1484 /* Write the grain directory entry to disk. */
1485 rc = vdIfIoIntFileWriteSync(pImage->pIfIo, pExtent->pFile->pStorage,
1486 VMDK_SECTOR2BYTE(pExtent->uSectorGD) + i * sizeof(uGTSectorLE),
1487 &uGTSectorLE, sizeof(uGTSectorLE));
1488 if (RT_FAILURE(rc))
1489 {
1490 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: cannot write new grain directory entry in '%s'"), pExtent->pszFullname);
1491 break;
1492 }
1493 uOffsetSectors += VMDK_BYTE2SECTOR(pExtent->cGTEntries * sizeof(uint32_t));
1494 }
1495 }
1496 }
1497 }
1498 }
1499
1500 if (RT_FAILURE(rc))
1501 vmdkFreeGrainDirectory(pExtent);
1502 return rc;
1503}
1504
1505/**
1506 * Unquotes the given string returning the result in a separate buffer.
1507 *
1508 * @returns VBox status code.
1509 * @param pImage The VMDK image state.
1510 * @param pszStr The string to unquote.
1511 * @param ppszUnquoted Where to store the return value, use RTMemTmpFree to
1512 * free.
1513 * @param ppszNext Where to store the pointer to any character following
1514 * the quoted value, optional.
1515 */
1516static int vmdkStringUnquote(PVMDKIMAGE pImage, const char *pszStr,
1517 char **ppszUnquoted, char **ppszNext)
1518{
1519 const char *pszStart = pszStr;
1520 char *pszQ;
1521 char *pszUnquoted;
1522
1523 /* Skip over whitespace. */
1524 while (*pszStr == ' ' || *pszStr == '\t')
1525 pszStr++;
1526
1527 if (*pszStr != '"')
1528 {
1529 pszQ = (char *)pszStr;
1530 while (*pszQ && *pszQ != ' ' && *pszQ != '\t')
1531 pszQ++;
1532 }
1533 else
1534 {
1535 pszStr++;
1536 pszQ = (char *)strchr(pszStr, '"');
1537 if (pszQ == NULL)
1538 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: incorrectly quoted value in descriptor in '%s' (raw value %s)"),
1539 pImage->pszFilename, pszStart);
1540 }
1541
1542 pszUnquoted = (char *)RTMemTmpAlloc(pszQ - pszStr + 1);
1543 if (!pszUnquoted)
1544 return VERR_NO_MEMORY;
1545 memcpy(pszUnquoted, pszStr, pszQ - pszStr);
1546 pszUnquoted[pszQ - pszStr] = '\0';
1547 *ppszUnquoted = pszUnquoted;
1548 if (ppszNext)
1549 *ppszNext = pszQ + 1;
1550 return VINF_SUCCESS;
1551}
1552
1553static int vmdkDescInitStr(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1554 const char *pszLine)
1555{
1556 char *pEnd = pDescriptor->aLines[pDescriptor->cLines];
1557 ssize_t cbDiff = strlen(pszLine) + 1;
1558
1559 if ( pDescriptor->cLines >= VMDK_DESCRIPTOR_LINES_MAX - 1
1560 && pEnd - pDescriptor->aLines[0] > (ptrdiff_t)pDescriptor->cbDescAlloc - cbDiff)
1561 return vdIfError(pImage->pIfError, VERR_BUFFER_OVERFLOW, RT_SRC_POS, N_("VMDK: descriptor too big in '%s'"), pImage->pszFilename);
1562
1563 memcpy(pEnd, pszLine, cbDiff);
1564 pDescriptor->cLines++;
1565 pDescriptor->aLines[pDescriptor->cLines] = pEnd + cbDiff;
1566 pDescriptor->fDirty = true;
1567
1568 return VINF_SUCCESS;
1569}
1570
1571static bool vmdkDescGetStr(PVMDKDESCRIPTOR pDescriptor, unsigned uStart,
1572 const char *pszKey, const char **ppszValue)
1573{
1574 size_t cbKey = strlen(pszKey);
1575 const char *pszValue;
1576
1577 while (uStart != 0)
1578 {
1579 if (!strncmp(pDescriptor->aLines[uStart], pszKey, cbKey))
1580 {
1581 /* Key matches, check for a '=' (preceded by whitespace). */
1582 pszValue = pDescriptor->aLines[uStart] + cbKey;
1583 while (*pszValue == ' ' || *pszValue == '\t')
1584 pszValue++;
1585 if (*pszValue == '=')
1586 {
1587 *ppszValue = pszValue + 1;
1588 break;
1589 }
1590 }
1591 uStart = pDescriptor->aNextLines[uStart];
1592 }
1593 return !!uStart;
1594}
1595
1596static int vmdkDescSetStr(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1597 unsigned uStart,
1598 const char *pszKey, const char *pszValue)
1599{
1600 char *pszTmp = NULL; /* (MSC naturally cannot figure this isn't used uninitialized) */
1601 size_t cbKey = strlen(pszKey);
1602 unsigned uLast = 0;
1603
1604 while (uStart != 0)
1605 {
1606 if (!strncmp(pDescriptor->aLines[uStart], pszKey, cbKey))
1607 {
1608 /* Key matches, check for a '=' (preceded by whitespace). */
1609 pszTmp = pDescriptor->aLines[uStart] + cbKey;
1610 while (*pszTmp == ' ' || *pszTmp == '\t')
1611 pszTmp++;
1612 if (*pszTmp == '=')
1613 {
1614 pszTmp++;
1615 /** @todo r=bird: Doesn't skipping trailing blanks here just cause unecessary
1616 * bloat and potentially out of space error? */
1617 while (*pszTmp == ' ' || *pszTmp == '\t')
1618 pszTmp++;
1619 break;
1620 }
1621 }
1622 if (!pDescriptor->aNextLines[uStart])
1623 uLast = uStart;
1624 uStart = pDescriptor->aNextLines[uStart];
1625 }
1626 if (uStart)
1627 {
1628 if (pszValue)
1629 {
1630 /* Key already exists, replace existing value. */
1631 size_t cbOldVal = strlen(pszTmp);
1632 size_t cbNewVal = strlen(pszValue);
1633 ssize_t cbDiff = cbNewVal - cbOldVal;
1634 /* Check for buffer overflow. */
1635 if ( pDescriptor->aLines[pDescriptor->cLines] - pDescriptor->aLines[0]
1636 > (ptrdiff_t)pDescriptor->cbDescAlloc - cbDiff)
1637 return vdIfError(pImage->pIfError, VERR_BUFFER_OVERFLOW, RT_SRC_POS, N_("VMDK: descriptor too big in '%s'"), pImage->pszFilename);
1638
1639 memmove(pszTmp + cbNewVal, pszTmp + cbOldVal,
1640 pDescriptor->aLines[pDescriptor->cLines] - pszTmp - cbOldVal);
1641 memcpy(pszTmp, pszValue, cbNewVal + 1);
1642 for (unsigned i = uStart + 1; i <= pDescriptor->cLines; i++)
1643 pDescriptor->aLines[i] += cbDiff;
1644 }
1645 else
1646 {
1647 memmove(pDescriptor->aLines[uStart], pDescriptor->aLines[uStart+1],
1648 pDescriptor->aLines[pDescriptor->cLines] - pDescriptor->aLines[uStart+1] + 1);
1649 for (unsigned i = uStart + 1; i <= pDescriptor->cLines; i++)
1650 {
1651 pDescriptor->aLines[i-1] = pDescriptor->aLines[i];
1652 if (pDescriptor->aNextLines[i])
1653 pDescriptor->aNextLines[i-1] = pDescriptor->aNextLines[i] - 1;
1654 else
1655 pDescriptor->aNextLines[i-1] = 0;
1656 }
1657 pDescriptor->cLines--;
1658 /* Adjust starting line numbers of following descriptor sections. */
1659 if (uStart < pDescriptor->uFirstExtent)
1660 pDescriptor->uFirstExtent--;
1661 if (uStart < pDescriptor->uFirstDDB)
1662 pDescriptor->uFirstDDB--;
1663 }
1664 }
1665 else
1666 {
1667 /* Key doesn't exist, append after the last entry in this category. */
1668 if (!pszValue)
1669 {
1670 /* Key doesn't exist, and it should be removed. Simply a no-op. */
1671 return VINF_SUCCESS;
1672 }
1673 cbKey = strlen(pszKey);
1674 size_t cbValue = strlen(pszValue);
1675 ssize_t cbDiff = cbKey + 1 + cbValue + 1;
1676 /* Check for buffer overflow. */
1677 if ( (pDescriptor->cLines >= VMDK_DESCRIPTOR_LINES_MAX - 1)
1678 || ( pDescriptor->aLines[pDescriptor->cLines]
1679 - pDescriptor->aLines[0] > (ptrdiff_t)pDescriptor->cbDescAlloc - cbDiff))
1680 return vdIfError(pImage->pIfError, VERR_BUFFER_OVERFLOW, RT_SRC_POS, N_("VMDK: descriptor too big in '%s'"), pImage->pszFilename);
1681 for (unsigned i = pDescriptor->cLines + 1; i > uLast + 1; i--)
1682 {
1683 pDescriptor->aLines[i] = pDescriptor->aLines[i - 1];
1684 if (pDescriptor->aNextLines[i - 1])
1685 pDescriptor->aNextLines[i] = pDescriptor->aNextLines[i - 1] + 1;
1686 else
1687 pDescriptor->aNextLines[i] = 0;
1688 }
1689 uStart = uLast + 1;
1690 pDescriptor->aNextLines[uLast] = uStart;
1691 pDescriptor->aNextLines[uStart] = 0;
1692 pDescriptor->cLines++;
1693 pszTmp = pDescriptor->aLines[uStart];
1694 memmove(pszTmp + cbDiff, pszTmp,
1695 pDescriptor->aLines[pDescriptor->cLines] - pszTmp);
1696 memcpy(pDescriptor->aLines[uStart], pszKey, cbKey);
1697 pDescriptor->aLines[uStart][cbKey] = '=';
1698 memcpy(pDescriptor->aLines[uStart] + cbKey + 1, pszValue, cbValue + 1);
1699 for (unsigned i = uStart + 1; i <= pDescriptor->cLines; i++)
1700 pDescriptor->aLines[i] += cbDiff;
1701
1702 /* Adjust starting line numbers of following descriptor sections. */
1703 if (uStart <= pDescriptor->uFirstExtent)
1704 pDescriptor->uFirstExtent++;
1705 if (uStart <= pDescriptor->uFirstDDB)
1706 pDescriptor->uFirstDDB++;
1707 }
1708 pDescriptor->fDirty = true;
1709 return VINF_SUCCESS;
1710}
1711
1712static int vmdkDescBaseGetU32(PVMDKDESCRIPTOR pDescriptor, const char *pszKey,
1713 uint32_t *puValue)
1714{
1715 const char *pszValue;
1716
1717 if (!vmdkDescGetStr(pDescriptor, pDescriptor->uFirstDesc, pszKey,
1718 &pszValue))
1719 return VERR_VD_VMDK_VALUE_NOT_FOUND;
1720 return RTStrToUInt32Ex(pszValue, NULL, 10, puValue);
1721}
1722
1723/**
1724 * Returns the value of the given key as a string allocating the necessary memory.
1725 *
1726 * @returns VBox status code.
1727 * @retval VERR_VD_VMDK_VALUE_NOT_FOUND if the value could not be found.
1728 * @param pImage The VMDK image state.
1729 * @param pDescriptor The descriptor to fetch the value from.
1730 * @param pszKey The key to get the value from.
1731 * @param ppszValue Where to store the return value, use RTMemTmpFree to
1732 * free.
1733 */
1734static int vmdkDescBaseGetStr(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1735 const char *pszKey, char **ppszValue)
1736{
1737 const char *pszValue;
1738 char *pszValueUnquoted;
1739
1740 if (!vmdkDescGetStr(pDescriptor, pDescriptor->uFirstDesc, pszKey,
1741 &pszValue))
1742 return VERR_VD_VMDK_VALUE_NOT_FOUND;
1743 int rc = vmdkStringUnquote(pImage, pszValue, &pszValueUnquoted, NULL);
1744 if (RT_FAILURE(rc))
1745 return rc;
1746 *ppszValue = pszValueUnquoted;
1747 return rc;
1748}
1749
1750static int vmdkDescBaseSetStr(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1751 const char *pszKey, const char *pszValue)
1752{
1753 char *pszValueQuoted;
1754
1755 RTStrAPrintf(&pszValueQuoted, "\"%s\"", pszValue);
1756 if (!pszValueQuoted)
1757 return VERR_NO_STR_MEMORY;
1758 int rc = vmdkDescSetStr(pImage, pDescriptor, pDescriptor->uFirstDesc, pszKey,
1759 pszValueQuoted);
1760 RTStrFree(pszValueQuoted);
1761 return rc;
1762}
1763
1764static void vmdkDescExtRemoveDummy(PVMDKIMAGE pImage,
1765 PVMDKDESCRIPTOR pDescriptor)
1766{
1767 RT_NOREF1(pImage);
1768 unsigned uEntry = pDescriptor->uFirstExtent;
1769 ssize_t cbDiff;
1770
1771 if (!uEntry)
1772 return;
1773
1774 cbDiff = strlen(pDescriptor->aLines[uEntry]) + 1;
1775 /* Move everything including \0 in the entry marking the end of buffer. */
1776 memmove(pDescriptor->aLines[uEntry], pDescriptor->aLines[uEntry + 1],
1777 pDescriptor->aLines[pDescriptor->cLines] - pDescriptor->aLines[uEntry + 1] + 1);
1778 for (unsigned i = uEntry + 1; i <= pDescriptor->cLines; i++)
1779 {
1780 pDescriptor->aLines[i - 1] = pDescriptor->aLines[i] - cbDiff;
1781 if (pDescriptor->aNextLines[i])
1782 pDescriptor->aNextLines[i - 1] = pDescriptor->aNextLines[i] - 1;
1783 else
1784 pDescriptor->aNextLines[i - 1] = 0;
1785 }
1786 pDescriptor->cLines--;
1787 if (pDescriptor->uFirstDDB)
1788 pDescriptor->uFirstDDB--;
1789
1790 return;
1791}
1792
1793static void vmdkDescExtRemoveByLine(PVMDKIMAGE pImage,
1794 PVMDKDESCRIPTOR pDescriptor, unsigned uLine)
1795{
1796 RT_NOREF1(pImage);
1797 unsigned uEntry = uLine;
1798 ssize_t cbDiff;
1799 if (!uEntry)
1800 return;
1801 cbDiff = strlen(pDescriptor->aLines[uEntry]) + 1;
1802 /* Move everything including \0 in the entry marking the end of buffer. */
1803 memmove(pDescriptor->aLines[uEntry], pDescriptor->aLines[uEntry + 1],
1804 pDescriptor->aLines[pDescriptor->cLines] - pDescriptor->aLines[uEntry + 1] + 1);
1805 for (unsigned i = uEntry; i <= pDescriptor->cLines; i++)
1806 {
1807 if (i != uEntry)
1808 pDescriptor->aLines[i - 1] = pDescriptor->aLines[i] - cbDiff;
1809 if (pDescriptor->aNextLines[i])
1810 pDescriptor->aNextLines[i - 1] = pDescriptor->aNextLines[i] - 1;
1811 else
1812 pDescriptor->aNextLines[i - 1] = 0;
1813 }
1814 pDescriptor->cLines--;
1815 if (pDescriptor->uFirstDDB)
1816 pDescriptor->uFirstDDB--;
1817 return;
1818}
1819
1820static int vmdkDescExtInsert(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1821 VMDKACCESS enmAccess, uint64_t cNominalSectors,
1822 VMDKETYPE enmType, const char *pszBasename,
1823 uint64_t uSectorOffset)
1824{
1825 static const char *apszAccess[] = { "NOACCESS", "RDONLY", "RW" };
1826 static const char *apszType[] = { "", "SPARSE", "FLAT", "ZERO", "VMFS" };
1827 char *pszTmp;
1828 unsigned uStart = pDescriptor->uFirstExtent, uLast = 0;
1829 char szExt[1024];
1830 ssize_t cbDiff;
1831
1832 Assert((unsigned)enmAccess < RT_ELEMENTS(apszAccess));
1833 Assert((unsigned)enmType < RT_ELEMENTS(apszType));
1834
1835 /* Find last entry in extent description. */
1836 while (uStart)
1837 {
1838 if (!pDescriptor->aNextLines[uStart])
1839 uLast = uStart;
1840 uStart = pDescriptor->aNextLines[uStart];
1841 }
1842
1843 if (enmType == VMDKETYPE_ZERO)
1844 {
1845 RTStrPrintf(szExt, sizeof(szExt), "%s %llu %s ", apszAccess[enmAccess],
1846 cNominalSectors, apszType[enmType]);
1847 }
1848 else if (enmType == VMDKETYPE_FLAT)
1849 {
1850 RTStrPrintf(szExt, sizeof(szExt), "%s %llu %s \"%s\" %llu",
1851 apszAccess[enmAccess], cNominalSectors,
1852 apszType[enmType], pszBasename, uSectorOffset);
1853 }
1854 else
1855 {
1856 RTStrPrintf(szExt, sizeof(szExt), "%s %llu %s \"%s\"",
1857 apszAccess[enmAccess], cNominalSectors,
1858 apszType[enmType], pszBasename);
1859 }
1860 cbDiff = strlen(szExt) + 1;
1861
1862 /* Check for buffer overflow. */
1863 if ( (pDescriptor->cLines >= VMDK_DESCRIPTOR_LINES_MAX - 1)
1864 || ( pDescriptor->aLines[pDescriptor->cLines]
1865 - pDescriptor->aLines[0] > (ptrdiff_t)pDescriptor->cbDescAlloc - cbDiff))
1866 {
1867 if ((pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_SPLIT_2G)
1868 && !(pDescriptor->cLines >= VMDK_DESCRIPTOR_LINES_MAX - 1))
1869 {
1870 pImage->cbDescAlloc *= 2;
1871 pDescriptor->cbDescAlloc *= 2;
1872 }
1873 else
1874 return vdIfError(pImage->pIfError, VERR_BUFFER_OVERFLOW, RT_SRC_POS, N_("VMDK: descriptor too big in '%s'"), pImage->pszFilename);
1875 }
1876
1877 for (unsigned i = pDescriptor->cLines + 1; i > uLast + 1; i--)
1878 {
1879 pDescriptor->aLines[i] = pDescriptor->aLines[i - 1];
1880 if (pDescriptor->aNextLines[i - 1])
1881 pDescriptor->aNextLines[i] = pDescriptor->aNextLines[i - 1] + 1;
1882 else
1883 pDescriptor->aNextLines[i] = 0;
1884 }
1885 uStart = uLast + 1;
1886 pDescriptor->aNextLines[uLast] = uStart;
1887 pDescriptor->aNextLines[uStart] = 0;
1888 pDescriptor->cLines++;
1889 pszTmp = pDescriptor->aLines[uStart];
1890 memmove(pszTmp + cbDiff, pszTmp,
1891 pDescriptor->aLines[pDescriptor->cLines] - pszTmp);
1892 memcpy(pDescriptor->aLines[uStart], szExt, cbDiff);
1893 for (unsigned i = uStart + 1; i <= pDescriptor->cLines; i++)
1894 pDescriptor->aLines[i] += cbDiff;
1895
1896 /* Adjust starting line numbers of following descriptor sections. */
1897 if (uStart <= pDescriptor->uFirstDDB)
1898 pDescriptor->uFirstDDB++;
1899
1900 pDescriptor->fDirty = true;
1901 return VINF_SUCCESS;
1902}
1903
1904/**
1905 * Returns the value of the given key from the DDB as a string allocating
1906 * the necessary memory.
1907 *
1908 * @returns VBox status code.
1909 * @retval VERR_VD_VMDK_VALUE_NOT_FOUND if the value could not be found.
1910 * @param pImage The VMDK image state.
1911 * @param pDescriptor The descriptor to fetch the value from.
1912 * @param pszKey The key to get the value from.
1913 * @param ppszValue Where to store the return value, use RTMemTmpFree to
1914 * free.
1915 */
1916static int vmdkDescDDBGetStr(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1917 const char *pszKey, char **ppszValue)
1918{
1919 const char *pszValue;
1920 char *pszValueUnquoted;
1921
1922 if (!vmdkDescGetStr(pDescriptor, pDescriptor->uFirstDDB, pszKey,
1923 &pszValue))
1924 return VERR_VD_VMDK_VALUE_NOT_FOUND;
1925 int rc = vmdkStringUnquote(pImage, pszValue, &pszValueUnquoted, NULL);
1926 if (RT_FAILURE(rc))
1927 return rc;
1928 *ppszValue = pszValueUnquoted;
1929 return rc;
1930}
1931
1932static int vmdkDescDDBGetU32(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1933 const char *pszKey, uint32_t *puValue)
1934{
1935 const char *pszValue;
1936 char *pszValueUnquoted;
1937
1938 if (!vmdkDescGetStr(pDescriptor, pDescriptor->uFirstDDB, pszKey,
1939 &pszValue))
1940 return VERR_VD_VMDK_VALUE_NOT_FOUND;
1941 int rc = vmdkStringUnquote(pImage, pszValue, &pszValueUnquoted, NULL);
1942 if (RT_FAILURE(rc))
1943 return rc;
1944 rc = RTStrToUInt32Ex(pszValueUnquoted, NULL, 10, puValue);
1945 RTMemTmpFree(pszValueUnquoted);
1946 return rc;
1947}
1948
1949static int vmdkDescDDBGetUuid(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1950 const char *pszKey, PRTUUID pUuid)
1951{
1952 const char *pszValue;
1953 char *pszValueUnquoted;
1954
1955 if (!vmdkDescGetStr(pDescriptor, pDescriptor->uFirstDDB, pszKey,
1956 &pszValue))
1957 return VERR_VD_VMDK_VALUE_NOT_FOUND;
1958 int rc = vmdkStringUnquote(pImage, pszValue, &pszValueUnquoted, NULL);
1959 if (RT_FAILURE(rc))
1960 return rc;
1961 rc = RTUuidFromStr(pUuid, pszValueUnquoted);
1962 RTMemTmpFree(pszValueUnquoted);
1963 return rc;
1964}
1965
1966static int vmdkDescDDBSetStr(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1967 const char *pszKey, const char *pszVal)
1968{
1969 int rc;
1970 char *pszValQuoted;
1971
1972 if (pszVal)
1973 {
1974 RTStrAPrintf(&pszValQuoted, "\"%s\"", pszVal);
1975 if (!pszValQuoted)
1976 return VERR_NO_STR_MEMORY;
1977 }
1978 else
1979 pszValQuoted = NULL;
1980 rc = vmdkDescSetStr(pImage, pDescriptor, pDescriptor->uFirstDDB, pszKey,
1981 pszValQuoted);
1982 if (pszValQuoted)
1983 RTStrFree(pszValQuoted);
1984 return rc;
1985}
1986
1987static int vmdkDescDDBSetUuid(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
1988 const char *pszKey, PCRTUUID pUuid)
1989{
1990 char *pszUuid;
1991
1992 RTStrAPrintf(&pszUuid, "\"%RTuuid\"", pUuid);
1993 if (!pszUuid)
1994 return VERR_NO_STR_MEMORY;
1995 int rc = vmdkDescSetStr(pImage, pDescriptor, pDescriptor->uFirstDDB, pszKey,
1996 pszUuid);
1997 RTStrFree(pszUuid);
1998 return rc;
1999}
2000
2001static int vmdkDescDDBSetU32(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDescriptor,
2002 const char *pszKey, uint32_t uValue)
2003{
2004 char *pszValue;
2005
2006 RTStrAPrintf(&pszValue, "\"%d\"", uValue);
2007 if (!pszValue)
2008 return VERR_NO_STR_MEMORY;
2009 int rc = vmdkDescSetStr(pImage, pDescriptor, pDescriptor->uFirstDDB, pszKey,
2010 pszValue);
2011 RTStrFree(pszValue);
2012 return rc;
2013}
2014
2015/**
2016 * Splits the descriptor data into individual lines checking for correct line
2017 * endings and descriptor size.
2018 *
2019 * @returns VBox status code.
2020 * @param pImage The image instance.
2021 * @param pDesc The descriptor.
2022 * @param pszTmp The raw descriptor data from the image.
2023 */
2024static int vmdkDescSplitLines(PVMDKIMAGE pImage, PVMDKDESCRIPTOR pDesc, char *pszTmp)
2025{
2026 unsigned cLine = 0;
2027 int rc = VINF_SUCCESS;
2028
2029 while ( RT_SUCCESS(rc)
2030 && *pszTmp != '\0')
2031 {
2032 pDesc->aLines[cLine++] = pszTmp;
2033 if (cLine >= VMDK_DESCRIPTOR_LINES_MAX)
2034 {
2035 vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: descriptor too big in '%s'"), pImage->pszFilename);
2036 rc = VERR_VD_VMDK_INVALID_HEADER;
2037 break;
2038 }
2039
2040 while (*pszTmp != '\0' && *pszTmp != '\n')
2041 {
2042 if (*pszTmp == '\r')
2043 {
2044 if (*(pszTmp + 1) != '\n')
2045 {
2046 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: unsupported end of line in descriptor in '%s'"), pImage->pszFilename);
2047 break;
2048 }
2049 else
2050 {
2051 /* Get rid of CR character. */
2052 *pszTmp = '\0';
2053 }
2054 }
2055 pszTmp++;
2056 }
2057
2058 if (RT_FAILURE(rc))
2059 break;
2060
2061 /* Get rid of LF character. */
2062 if (*pszTmp == '\n')
2063 {
2064 *pszTmp = '\0';
2065 pszTmp++;
2066 }
2067 }
2068
2069 if (RT_SUCCESS(rc))
2070 {
2071 pDesc->cLines = cLine;
2072 /* Pointer right after the end of the used part of the buffer. */
2073 pDesc->aLines[cLine] = pszTmp;
2074 }
2075
2076 return rc;
2077}
2078
2079static int vmdkPreprocessDescriptor(PVMDKIMAGE pImage, char *pDescData,
2080 size_t cbDescData, PVMDKDESCRIPTOR pDescriptor)
2081{
2082 pDescriptor->cbDescAlloc = cbDescData;
2083 int rc = vmdkDescSplitLines(pImage, pDescriptor, pDescData);
2084 if (RT_SUCCESS(rc))
2085 {
2086 if ( strcmp(pDescriptor->aLines[0], "# Disk DescriptorFile")
2087 && strcmp(pDescriptor->aLines[0], "# Disk Descriptor File")
2088 && strcmp(pDescriptor->aLines[0], "#Disk Descriptor File")
2089 && strcmp(pDescriptor->aLines[0], "#Disk DescriptorFile"))
2090 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2091 N_("VMDK: descriptor does not start as expected in '%s'"), pImage->pszFilename);
2092 else
2093 {
2094 unsigned uLastNonEmptyLine = 0;
2095
2096 /* Initialize those, because we need to be able to reopen an image. */
2097 pDescriptor->uFirstDesc = 0;
2098 pDescriptor->uFirstExtent = 0;
2099 pDescriptor->uFirstDDB = 0;
2100 for (unsigned i = 0; i < pDescriptor->cLines; i++)
2101 {
2102 if (*pDescriptor->aLines[i] != '#' && *pDescriptor->aLines[i] != '\0')
2103 {
2104 if ( !strncmp(pDescriptor->aLines[i], "RW", 2)
2105 || !strncmp(pDescriptor->aLines[i], "RDONLY", 6)
2106 || !strncmp(pDescriptor->aLines[i], "NOACCESS", 8) )
2107 {
2108 /* An extent descriptor. */
2109 if (!pDescriptor->uFirstDesc || pDescriptor->uFirstDDB)
2110 {
2111 /* Incorrect ordering of entries. */
2112 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2113 N_("VMDK: incorrect ordering of entries in descriptor in '%s'"), pImage->pszFilename);
2114 break;
2115 }
2116 if (!pDescriptor->uFirstExtent)
2117 {
2118 pDescriptor->uFirstExtent = i;
2119 uLastNonEmptyLine = 0;
2120 }
2121 }
2122 else if (!strncmp(pDescriptor->aLines[i], "ddb.", 4))
2123 {
2124 /* A disk database entry. */
2125 if (!pDescriptor->uFirstDesc || !pDescriptor->uFirstExtent)
2126 {
2127 /* Incorrect ordering of entries. */
2128 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2129 N_("VMDK: incorrect ordering of entries in descriptor in '%s'"), pImage->pszFilename);
2130 break;
2131 }
2132 if (!pDescriptor->uFirstDDB)
2133 {
2134 pDescriptor->uFirstDDB = i;
2135 uLastNonEmptyLine = 0;
2136 }
2137 }
2138 else
2139 {
2140 /* A normal entry. */
2141 if (pDescriptor->uFirstExtent || pDescriptor->uFirstDDB)
2142 {
2143 /* Incorrect ordering of entries. */
2144 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2145 N_("VMDK: incorrect ordering of entries in descriptor in '%s'"), pImage->pszFilename);
2146 break;
2147 }
2148 if (!pDescriptor->uFirstDesc)
2149 {
2150 pDescriptor->uFirstDesc = i;
2151 uLastNonEmptyLine = 0;
2152 }
2153 }
2154 if (uLastNonEmptyLine)
2155 pDescriptor->aNextLines[uLastNonEmptyLine] = i;
2156 uLastNonEmptyLine = i;
2157 }
2158 }
2159 }
2160 }
2161
2162 return rc;
2163}
2164
2165static int vmdkDescSetPCHSGeometry(PVMDKIMAGE pImage,
2166 PCVDGEOMETRY pPCHSGeometry)
2167{
2168 int rc = vmdkDescDDBSetU32(pImage, &pImage->Descriptor,
2169 VMDK_DDB_GEO_PCHS_CYLINDERS,
2170 pPCHSGeometry->cCylinders);
2171 if (RT_FAILURE(rc))
2172 return rc;
2173 rc = vmdkDescDDBSetU32(pImage, &pImage->Descriptor,
2174 VMDK_DDB_GEO_PCHS_HEADS,
2175 pPCHSGeometry->cHeads);
2176 if (RT_FAILURE(rc))
2177 return rc;
2178 rc = vmdkDescDDBSetU32(pImage, &pImage->Descriptor,
2179 VMDK_DDB_GEO_PCHS_SECTORS,
2180 pPCHSGeometry->cSectors);
2181 return rc;
2182}
2183
2184static int vmdkDescSetLCHSGeometry(PVMDKIMAGE pImage,
2185 PCVDGEOMETRY pLCHSGeometry)
2186{
2187 int rc = vmdkDescDDBSetU32(pImage, &pImage->Descriptor,
2188 VMDK_DDB_GEO_LCHS_CYLINDERS,
2189 pLCHSGeometry->cCylinders);
2190 if (RT_FAILURE(rc))
2191 return rc;
2192 rc = vmdkDescDDBSetU32(pImage, &pImage->Descriptor,
2193 VMDK_DDB_GEO_LCHS_HEADS,
2194
2195 pLCHSGeometry->cHeads);
2196 if (RT_FAILURE(rc))
2197 return rc;
2198 rc = vmdkDescDDBSetU32(pImage, &pImage->Descriptor,
2199 VMDK_DDB_GEO_LCHS_SECTORS,
2200 pLCHSGeometry->cSectors);
2201 return rc;
2202}
2203
2204static int vmdkCreateDescriptor(PVMDKIMAGE pImage, char *pDescData,
2205 size_t cbDescData, PVMDKDESCRIPTOR pDescriptor)
2206{
2207 pDescriptor->uFirstDesc = 0;
2208 pDescriptor->uFirstExtent = 0;
2209 pDescriptor->uFirstDDB = 0;
2210 pDescriptor->cLines = 0;
2211 pDescriptor->cbDescAlloc = cbDescData;
2212 pDescriptor->fDirty = false;
2213 pDescriptor->aLines[pDescriptor->cLines] = pDescData;
2214 memset(pDescriptor->aNextLines, '\0', sizeof(pDescriptor->aNextLines));
2215
2216 int rc = vmdkDescInitStr(pImage, pDescriptor, "# Disk DescriptorFile");
2217 if (RT_SUCCESS(rc))
2218 rc = vmdkDescInitStr(pImage, pDescriptor, "version=1");
2219 if (RT_SUCCESS(rc))
2220 {
2221 pDescriptor->uFirstDesc = pDescriptor->cLines - 1;
2222 rc = vmdkDescInitStr(pImage, pDescriptor, "");
2223 }
2224 if (RT_SUCCESS(rc))
2225 rc = vmdkDescInitStr(pImage, pDescriptor, "# Extent description");
2226 if (RT_SUCCESS(rc))
2227 rc = vmdkDescInitStr(pImage, pDescriptor, "NOACCESS 0 ZERO ");
2228 if (RT_SUCCESS(rc))
2229 {
2230 pDescriptor->uFirstExtent = pDescriptor->cLines - 1;
2231 rc = vmdkDescInitStr(pImage, pDescriptor, "");
2232 }
2233 if (RT_SUCCESS(rc))
2234 {
2235 /* The trailing space is created by VMware, too. */
2236 rc = vmdkDescInitStr(pImage, pDescriptor, "# The disk Data Base ");
2237 }
2238 if (RT_SUCCESS(rc))
2239 rc = vmdkDescInitStr(pImage, pDescriptor, "#DDB");
2240 if (RT_SUCCESS(rc))
2241 rc = vmdkDescInitStr(pImage, pDescriptor, "");
2242 if (RT_SUCCESS(rc))
2243 rc = vmdkDescInitStr(pImage, pDescriptor, "ddb.virtualHWVersion = \"4\"");
2244 if (RT_SUCCESS(rc))
2245 {
2246 pDescriptor->uFirstDDB = pDescriptor->cLines - 1;
2247
2248 /* Now that the framework is in place, use the normal functions to insert
2249 * the remaining keys. */
2250 char szBuf[9];
2251 RTStrPrintf(szBuf, sizeof(szBuf), "%08x", RTRandU32());
2252 rc = vmdkDescSetStr(pImage, pDescriptor, pDescriptor->uFirstDesc,
2253 "CID", szBuf);
2254 }
2255 if (RT_SUCCESS(rc))
2256 rc = vmdkDescSetStr(pImage, pDescriptor, pDescriptor->uFirstDesc,
2257 "parentCID", "ffffffff");
2258 if (RT_SUCCESS(rc))
2259 rc = vmdkDescDDBSetStr(pImage, pDescriptor, "ddb.adapterType", "ide");
2260
2261 return rc;
2262}
2263
2264static int vmdkParseDescriptor(PVMDKIMAGE pImage, char *pDescData, size_t cbDescData)
2265{
2266 int rc;
2267 unsigned cExtents;
2268 unsigned uLine;
2269 unsigned i;
2270
2271 rc = vmdkPreprocessDescriptor(pImage, pDescData, cbDescData,
2272 &pImage->Descriptor);
2273 if (RT_FAILURE(rc))
2274 return rc;
2275
2276 /* Check version, must be 1. */
2277 uint32_t uVersion;
2278 rc = vmdkDescBaseGetU32(&pImage->Descriptor, "version", &uVersion);
2279 if (RT_FAILURE(rc))
2280 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error finding key 'version' in descriptor in '%s'"), pImage->pszFilename);
2281 if (uVersion != 1)
2282 return vdIfError(pImage->pIfError, VERR_VD_VMDK_UNSUPPORTED_VERSION, RT_SRC_POS, N_("VMDK: unsupported format version in descriptor in '%s'"), pImage->pszFilename);
2283
2284 /* Get image creation type and determine image flags. */
2285 char *pszCreateType = NULL; /* initialized to make gcc shut up */
2286 rc = vmdkDescBaseGetStr(pImage, &pImage->Descriptor, "createType",
2287 &pszCreateType);
2288 if (RT_FAILURE(rc))
2289 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: cannot get image type from descriptor in '%s'"), pImage->pszFilename);
2290 if ( !strcmp(pszCreateType, "twoGbMaxExtentSparse")
2291 || !strcmp(pszCreateType, "twoGbMaxExtentFlat"))
2292 pImage->uImageFlags |= VD_VMDK_IMAGE_FLAGS_SPLIT_2G;
2293 else if ( !strcmp(pszCreateType, "partitionedDevice")
2294 || !strcmp(pszCreateType, "fullDevice"))
2295 pImage->uImageFlags |= VD_VMDK_IMAGE_FLAGS_RAWDISK;
2296 else if (!strcmp(pszCreateType, "streamOptimized"))
2297 pImage->uImageFlags |= VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED;
2298 else if (!strcmp(pszCreateType, "vmfs"))
2299 pImage->uImageFlags |= VD_IMAGE_FLAGS_FIXED | VD_VMDK_IMAGE_FLAGS_ESX;
2300 RTMemTmpFree(pszCreateType);
2301
2302 /* Count the number of extent config entries. */
2303 for (uLine = pImage->Descriptor.uFirstExtent, cExtents = 0;
2304 uLine != 0;
2305 uLine = pImage->Descriptor.aNextLines[uLine], cExtents++)
2306 /* nothing */;
2307
2308 if (!pImage->pDescData && cExtents != 1)
2309 {
2310 /* Monolithic image, must have only one extent (already opened). */
2311 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: monolithic image may only have one extent in '%s'"), pImage->pszFilename);
2312 }
2313
2314 if (pImage->pDescData)
2315 {
2316 /* Non-monolithic image, extents need to be allocated. */
2317 rc = vmdkCreateExtents(pImage, cExtents);
2318 if (RT_FAILURE(rc))
2319 return rc;
2320 }
2321
2322 for (i = 0, uLine = pImage->Descriptor.uFirstExtent;
2323 i < cExtents; i++, uLine = pImage->Descriptor.aNextLines[uLine])
2324 {
2325 char *pszLine = pImage->Descriptor.aLines[uLine];
2326
2327 /* Access type of the extent. */
2328 if (!strncmp(pszLine, "RW", 2))
2329 {
2330 pImage->pExtents[i].enmAccess = VMDKACCESS_READWRITE;
2331 pszLine += 2;
2332 }
2333 else if (!strncmp(pszLine, "RDONLY", 6))
2334 {
2335 pImage->pExtents[i].enmAccess = VMDKACCESS_READONLY;
2336 pszLine += 6;
2337 }
2338 else if (!strncmp(pszLine, "NOACCESS", 8))
2339 {
2340 pImage->pExtents[i].enmAccess = VMDKACCESS_NOACCESS;
2341 pszLine += 8;
2342 }
2343 else
2344 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: parse error in extent description in '%s'"), pImage->pszFilename);
2345 if (*pszLine++ != ' ')
2346 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: parse error in extent description in '%s'"), pImage->pszFilename);
2347
2348 /* Nominal size of the extent. */
2349 rc = RTStrToUInt64Ex(pszLine, &pszLine, 10,
2350 &pImage->pExtents[i].cNominalSectors);
2351 if (RT_FAILURE(rc))
2352 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: parse error in extent description in '%s'"), pImage->pszFilename);
2353 if (*pszLine++ != ' ')
2354 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: parse error in extent description in '%s'"), pImage->pszFilename);
2355
2356 /* Type of the extent. */
2357 if (!strncmp(pszLine, "SPARSE", 6))
2358 {
2359 pImage->pExtents[i].enmType = VMDKETYPE_HOSTED_SPARSE;
2360 pszLine += 6;
2361 }
2362 else if (!strncmp(pszLine, "FLAT", 4))
2363 {
2364 pImage->pExtents[i].enmType = VMDKETYPE_FLAT;
2365 pszLine += 4;
2366 }
2367 else if (!strncmp(pszLine, "ZERO", 4))
2368 {
2369 pImage->pExtents[i].enmType = VMDKETYPE_ZERO;
2370 pszLine += 4;
2371 }
2372 else if (!strncmp(pszLine, "VMFS", 4))
2373 {
2374 pImage->pExtents[i].enmType = VMDKETYPE_VMFS;
2375 pszLine += 4;
2376 }
2377 else
2378 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: parse error in extent description in '%s'"), pImage->pszFilename);
2379
2380 if (pImage->pExtents[i].enmType == VMDKETYPE_ZERO)
2381 {
2382 /* This one has no basename or offset. */
2383 if (*pszLine == ' ')
2384 pszLine++;
2385 if (*pszLine != '\0')
2386 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: parse error in extent description in '%s'"), pImage->pszFilename);
2387 pImage->pExtents[i].pszBasename = NULL;
2388 }
2389 else
2390 {
2391 /* All other extent types have basename and optional offset. */
2392 if (*pszLine++ != ' ')
2393 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: parse error in extent description in '%s'"), pImage->pszFilename);
2394
2395 /* Basename of the image. Surrounded by quotes. */
2396 char *pszBasename;
2397 rc = vmdkStringUnquote(pImage, pszLine, &pszBasename, &pszLine);
2398 if (RT_FAILURE(rc))
2399 return rc;
2400 pImage->pExtents[i].pszBasename = pszBasename;
2401 if (*pszLine == ' ')
2402 {
2403 pszLine++;
2404 if (*pszLine != '\0')
2405 {
2406 /* Optional offset in extent specified. */
2407 rc = RTStrToUInt64Ex(pszLine, &pszLine, 10,
2408 &pImage->pExtents[i].uSectorOffset);
2409 if (RT_FAILURE(rc))
2410 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: parse error in extent description in '%s'"), pImage->pszFilename);
2411 }
2412 }
2413
2414 if (*pszLine != '\0')
2415 return vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: parse error in extent description in '%s'"), pImage->pszFilename);
2416 }
2417 }
2418
2419 /* Determine PCHS geometry (autogenerate if necessary). */
2420 rc = vmdkDescDDBGetU32(pImage, &pImage->Descriptor,
2421 VMDK_DDB_GEO_PCHS_CYLINDERS,
2422 &pImage->PCHSGeometry.cCylinders);
2423 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2424 pImage->PCHSGeometry.cCylinders = 0;
2425 else if (RT_FAILURE(rc))
2426 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error getting PCHS geometry from extent description in '%s'"), pImage->pszFilename);
2427 rc = vmdkDescDDBGetU32(pImage, &pImage->Descriptor,
2428 VMDK_DDB_GEO_PCHS_HEADS,
2429 &pImage->PCHSGeometry.cHeads);
2430 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2431 pImage->PCHSGeometry.cHeads = 0;
2432 else if (RT_FAILURE(rc))
2433 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error getting PCHS geometry from extent description in '%s'"), pImage->pszFilename);
2434 rc = vmdkDescDDBGetU32(pImage, &pImage->Descriptor,
2435 VMDK_DDB_GEO_PCHS_SECTORS,
2436 &pImage->PCHSGeometry.cSectors);
2437 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2438 pImage->PCHSGeometry.cSectors = 0;
2439 else if (RT_FAILURE(rc))
2440 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error getting PCHS geometry from extent description in '%s'"), pImage->pszFilename);
2441 if ( pImage->PCHSGeometry.cCylinders == 0
2442 || pImage->PCHSGeometry.cHeads == 0
2443 || pImage->PCHSGeometry.cHeads > 16
2444 || pImage->PCHSGeometry.cSectors == 0
2445 || pImage->PCHSGeometry.cSectors > 63)
2446 {
2447 /* Mark PCHS geometry as not yet valid (can't do the calculation here
2448 * as the total image size isn't known yet). */
2449 pImage->PCHSGeometry.cCylinders = 0;
2450 pImage->PCHSGeometry.cHeads = 16;
2451 pImage->PCHSGeometry.cSectors = 63;
2452 }
2453
2454 /* Determine LCHS geometry (set to 0 if not specified). */
2455 rc = vmdkDescDDBGetU32(pImage, &pImage->Descriptor,
2456 VMDK_DDB_GEO_LCHS_CYLINDERS,
2457 &pImage->LCHSGeometry.cCylinders);
2458 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2459 pImage->LCHSGeometry.cCylinders = 0;
2460 else if (RT_FAILURE(rc))
2461 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error getting LCHS geometry from extent description in '%s'"), pImage->pszFilename);
2462 rc = vmdkDescDDBGetU32(pImage, &pImage->Descriptor,
2463 VMDK_DDB_GEO_LCHS_HEADS,
2464 &pImage->LCHSGeometry.cHeads);
2465 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2466 pImage->LCHSGeometry.cHeads = 0;
2467 else if (RT_FAILURE(rc))
2468 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error getting LCHS geometry from extent description in '%s'"), pImage->pszFilename);
2469 rc = vmdkDescDDBGetU32(pImage, &pImage->Descriptor,
2470 VMDK_DDB_GEO_LCHS_SECTORS,
2471 &pImage->LCHSGeometry.cSectors);
2472 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2473 pImage->LCHSGeometry.cSectors = 0;
2474 else if (RT_FAILURE(rc))
2475 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error getting LCHS geometry from extent description in '%s'"), pImage->pszFilename);
2476 if ( pImage->LCHSGeometry.cCylinders == 0
2477 || pImage->LCHSGeometry.cHeads == 0
2478 || pImage->LCHSGeometry.cSectors == 0)
2479 {
2480 pImage->LCHSGeometry.cCylinders = 0;
2481 pImage->LCHSGeometry.cHeads = 0;
2482 pImage->LCHSGeometry.cSectors = 0;
2483 }
2484
2485 /* Get image UUID. */
2486 rc = vmdkDescDDBGetUuid(pImage, &pImage->Descriptor, VMDK_DDB_IMAGE_UUID,
2487 &pImage->ImageUuid);
2488 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2489 {
2490 /* Image without UUID. Probably created by VMware and not yet used
2491 * by VirtualBox. Can only be added for images opened in read/write
2492 * mode, so don't bother producing a sensible UUID otherwise. */
2493 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2494 RTUuidClear(&pImage->ImageUuid);
2495 else
2496 {
2497 rc = RTUuidCreate(&pImage->ImageUuid);
2498 if (RT_FAILURE(rc))
2499 return rc;
2500 rc = vmdkDescDDBSetUuid(pImage, &pImage->Descriptor,
2501 VMDK_DDB_IMAGE_UUID, &pImage->ImageUuid);
2502 if (RT_FAILURE(rc))
2503 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error storing image UUID in descriptor in '%s'"), pImage->pszFilename);
2504 }
2505 }
2506 else if (RT_FAILURE(rc))
2507 return rc;
2508
2509 /* Get image modification UUID. */
2510 rc = vmdkDescDDBGetUuid(pImage, &pImage->Descriptor,
2511 VMDK_DDB_MODIFICATION_UUID,
2512 &pImage->ModificationUuid);
2513 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2514 {
2515 /* Image without UUID. Probably created by VMware and not yet used
2516 * by VirtualBox. Can only be added for images opened in read/write
2517 * mode, so don't bother producing a sensible UUID otherwise. */
2518 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2519 RTUuidClear(&pImage->ModificationUuid);
2520 else
2521 {
2522 rc = RTUuidCreate(&pImage->ModificationUuid);
2523 if (RT_FAILURE(rc))
2524 return rc;
2525 rc = vmdkDescDDBSetUuid(pImage, &pImage->Descriptor,
2526 VMDK_DDB_MODIFICATION_UUID,
2527 &pImage->ModificationUuid);
2528 if (RT_FAILURE(rc))
2529 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error storing image modification UUID in descriptor in '%s'"), pImage->pszFilename);
2530 }
2531 }
2532 else if (RT_FAILURE(rc))
2533 return rc;
2534
2535 /* Get UUID of parent image. */
2536 rc = vmdkDescDDBGetUuid(pImage, &pImage->Descriptor, VMDK_DDB_PARENT_UUID,
2537 &pImage->ParentUuid);
2538 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2539 {
2540 /* Image without UUID. Probably created by VMware and not yet used
2541 * by VirtualBox. Can only be added for images opened in read/write
2542 * mode, so don't bother producing a sensible UUID otherwise. */
2543 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2544 RTUuidClear(&pImage->ParentUuid);
2545 else
2546 {
2547 rc = RTUuidClear(&pImage->ParentUuid);
2548 if (RT_FAILURE(rc))
2549 return rc;
2550 rc = vmdkDescDDBSetUuid(pImage, &pImage->Descriptor,
2551 VMDK_DDB_PARENT_UUID, &pImage->ParentUuid);
2552 if (RT_FAILURE(rc))
2553 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error storing parent UUID in descriptor in '%s'"), pImage->pszFilename);
2554 }
2555 }
2556 else if (RT_FAILURE(rc))
2557 return rc;
2558
2559 /* Get parent image modification UUID. */
2560 rc = vmdkDescDDBGetUuid(pImage, &pImage->Descriptor,
2561 VMDK_DDB_PARENT_MODIFICATION_UUID,
2562 &pImage->ParentModificationUuid);
2563 if (rc == VERR_VD_VMDK_VALUE_NOT_FOUND)
2564 {
2565 /* Image without UUID. Probably created by VMware and not yet used
2566 * by VirtualBox. Can only be added for images opened in read/write
2567 * mode, so don't bother producing a sensible UUID otherwise. */
2568 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2569 RTUuidClear(&pImage->ParentModificationUuid);
2570 else
2571 {
2572 RTUuidClear(&pImage->ParentModificationUuid);
2573 rc = vmdkDescDDBSetUuid(pImage, &pImage->Descriptor,
2574 VMDK_DDB_PARENT_MODIFICATION_UUID,
2575 &pImage->ParentModificationUuid);
2576 if (RT_FAILURE(rc))
2577 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error storing parent modification UUID in descriptor in '%s'"), pImage->pszFilename);
2578 }
2579 }
2580 else if (RT_FAILURE(rc))
2581 return rc;
2582
2583 return VINF_SUCCESS;
2584}
2585
2586/**
2587 * Internal : Prepares the descriptor to write to the image.
2588 */
2589static int vmdkDescriptorPrepare(PVMDKIMAGE pImage, uint64_t cbLimit,
2590 void **ppvData, size_t *pcbData)
2591{
2592 int rc = VINF_SUCCESS;
2593
2594 /*
2595 * Allocate temporary descriptor buffer.
2596 * In case there is no limit allocate a default
2597 * and increase if required.
2598 */
2599 size_t cbDescriptor = cbLimit ? cbLimit : 4 * _1K;
2600 char *pszDescriptor = (char *)RTMemAllocZ(cbDescriptor);
2601 size_t offDescriptor = 0;
2602
2603 if (!pszDescriptor)
2604 return VERR_NO_MEMORY;
2605
2606 for (unsigned i = 0; i < pImage->Descriptor.cLines; i++)
2607 {
2608 const char *psz = pImage->Descriptor.aLines[i];
2609 size_t cb = strlen(psz);
2610
2611 /*
2612 * Increase the descriptor if there is no limit and
2613 * there is not enough room left for this line.
2614 */
2615 if (offDescriptor + cb + 1 > cbDescriptor)
2616 {
2617 if (cbLimit)
2618 {
2619 rc = vdIfError(pImage->pIfError, VERR_BUFFER_OVERFLOW, RT_SRC_POS, N_("VMDK: descriptor too long in '%s'"), pImage->pszFilename);
2620 break;
2621 }
2622 else
2623 {
2624 char *pszDescriptorNew = NULL;
2625 LogFlow(("Increasing descriptor cache\n"));
2626
2627 pszDescriptorNew = (char *)RTMemRealloc(pszDescriptor, cbDescriptor + cb + 4 * _1K);
2628 if (!pszDescriptorNew)
2629 {
2630 rc = VERR_NO_MEMORY;
2631 break;
2632 }
2633 pszDescriptor = pszDescriptorNew;
2634 cbDescriptor += cb + 4 * _1K;
2635 }
2636 }
2637
2638 if (cb > 0)
2639 {
2640 memcpy(pszDescriptor + offDescriptor, psz, cb);
2641 offDescriptor += cb;
2642 }
2643
2644 memcpy(pszDescriptor + offDescriptor, "\n", 1);
2645 offDescriptor++;
2646 }
2647
2648 if (RT_SUCCESS(rc))
2649 {
2650 *ppvData = pszDescriptor;
2651 *pcbData = offDescriptor;
2652 }
2653 else if (pszDescriptor)
2654 RTMemFree(pszDescriptor);
2655
2656 return rc;
2657}
2658
2659/**
2660 * Internal: write/update the descriptor part of the image.
2661 */
2662static int vmdkWriteDescriptor(PVMDKIMAGE pImage, PVDIOCTX pIoCtx)
2663{
2664 int rc = VINF_SUCCESS;
2665 uint64_t cbLimit;
2666 uint64_t uOffset;
2667 PVMDKFILE pDescFile;
2668 void *pvDescriptor = NULL;
2669 size_t cbDescriptor;
2670
2671 if (pImage->pDescData)
2672 {
2673 /* Separate descriptor file. */
2674 uOffset = 0;
2675 cbLimit = 0;
2676 pDescFile = pImage->pFile;
2677 }
2678 else
2679 {
2680 /* Embedded descriptor file. */
2681 uOffset = VMDK_SECTOR2BYTE(pImage->pExtents[0].uDescriptorSector);
2682 cbLimit = VMDK_SECTOR2BYTE(pImage->pExtents[0].cDescriptorSectors);
2683 pDescFile = pImage->pExtents[0].pFile;
2684 }
2685 /* Bail out if there is no file to write to. */
2686 if (pDescFile == NULL)
2687 return VERR_INVALID_PARAMETER;
2688
2689 rc = vmdkDescriptorPrepare(pImage, cbLimit, &pvDescriptor, &cbDescriptor);
2690 if (RT_SUCCESS(rc))
2691 {
2692 rc = vdIfIoIntFileWriteMeta(pImage->pIfIo, pDescFile->pStorage,
2693 uOffset, pvDescriptor,
2694 cbLimit ? cbLimit : cbDescriptor,
2695 pIoCtx, NULL, NULL);
2696 if ( RT_FAILURE(rc)
2697 && rc != VERR_VD_ASYNC_IO_IN_PROGRESS)
2698 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error writing descriptor in '%s'"), pImage->pszFilename);
2699 }
2700
2701 if (RT_SUCCESS(rc) && !cbLimit)
2702 {
2703 rc = vdIfIoIntFileSetSize(pImage->pIfIo, pDescFile->pStorage, cbDescriptor);
2704 if (RT_FAILURE(rc))
2705 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error truncating descriptor in '%s'"), pImage->pszFilename);
2706 }
2707
2708 if (RT_SUCCESS(rc))
2709 pImage->Descriptor.fDirty = false;
2710
2711 if (pvDescriptor)
2712 RTMemFree(pvDescriptor);
2713 return rc;
2714
2715}
2716
2717/**
2718 * Internal: validate the consistency check values in a binary header.
2719 */
2720static int vmdkValidateHeader(PVMDKIMAGE pImage, PVMDKEXTENT pExtent, const SparseExtentHeader *pHeader)
2721{
2722 int rc = VINF_SUCCESS;
2723 if (RT_LE2H_U32(pHeader->magicNumber) != VMDK_SPARSE_MAGICNUMBER)
2724 {
2725 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: incorrect magic in sparse extent header in '%s'"), pExtent->pszFullname);
2726 return rc;
2727 }
2728 if (RT_LE2H_U32(pHeader->version) != 1 && RT_LE2H_U32(pHeader->version) != 3)
2729 {
2730 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_UNSUPPORTED_VERSION, RT_SRC_POS, N_("VMDK: incorrect version in sparse extent header in '%s', not a VMDK 1.0/1.1 conforming file"), pExtent->pszFullname);
2731 return rc;
2732 }
2733 if ( (RT_LE2H_U32(pHeader->flags) & 1)
2734 && ( pHeader->singleEndLineChar != '\n'
2735 || pHeader->nonEndLineChar != ' '
2736 || pHeader->doubleEndLineChar1 != '\r'
2737 || pHeader->doubleEndLineChar2 != '\n') )
2738 {
2739 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: corrupted by CR/LF translation in '%s'"), pExtent->pszFullname);
2740 return rc;
2741 }
2742 if (RT_LE2H_U64(pHeader->descriptorSize) > VMDK_SPARSE_DESCRIPTOR_SIZE_MAX)
2743 {
2744 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: descriptor size out of bounds (%llu vs %llu) '%s'"),
2745 pExtent->pszFullname, RT_LE2H_U64(pHeader->descriptorSize), VMDK_SPARSE_DESCRIPTOR_SIZE_MAX);
2746 return rc;
2747 }
2748 return rc;
2749}
2750
2751/**
2752 * Internal: read metadata belonging to an extent with binary header, i.e.
2753 * as found in monolithic files.
2754 */
2755static int vmdkReadBinaryMetaExtent(PVMDKIMAGE pImage, PVMDKEXTENT pExtent,
2756 bool fMagicAlreadyRead)
2757{
2758 SparseExtentHeader Header;
2759 int rc;
2760
2761 if (!fMagicAlreadyRead)
2762 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage, 0,
2763 &Header, sizeof(Header));
2764 else
2765 {
2766 Header.magicNumber = RT_H2LE_U32(VMDK_SPARSE_MAGICNUMBER);
2767 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage,
2768 RT_UOFFSETOF(SparseExtentHeader, version),
2769 &Header.version,
2770 sizeof(Header)
2771 - RT_UOFFSETOF(SparseExtentHeader, version));
2772 }
2773
2774 if (RT_SUCCESS(rc))
2775 {
2776 rc = vmdkValidateHeader(pImage, pExtent, &Header);
2777 if (RT_SUCCESS(rc))
2778 {
2779 uint64_t cbFile = 0;
2780
2781 if ( (RT_LE2H_U32(Header.flags) & RT_BIT(17))
2782 && RT_LE2H_U64(Header.gdOffset) == VMDK_GD_AT_END)
2783 pExtent->fFooter = true;
2784
2785 if ( !(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2786 || ( pExtent->fFooter
2787 && !(pImage->uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL)))
2788 {
2789 rc = vdIfIoIntFileGetSize(pImage->pIfIo, pExtent->pFile->pStorage, &cbFile);
2790 if (RT_FAILURE(rc))
2791 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: cannot get size of '%s'"), pExtent->pszFullname);
2792 }
2793
2794 if (RT_SUCCESS(rc))
2795 {
2796 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
2797 pExtent->uAppendPosition = RT_ALIGN_64(cbFile, 512);
2798
2799 if ( pExtent->fFooter
2800 && ( !(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2801 || !(pImage->uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL)))
2802 {
2803 /* Read the footer, which comes before the end-of-stream marker. */
2804 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage,
2805 cbFile - 2*512, &Header,
2806 sizeof(Header));
2807 if (RT_FAILURE(rc))
2808 {
2809 vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error reading extent footer in '%s'"), pExtent->pszFullname);
2810 rc = VERR_VD_VMDK_INVALID_HEADER;
2811 }
2812
2813 if (RT_SUCCESS(rc))
2814 rc = vmdkValidateHeader(pImage, pExtent, &Header);
2815 /* Prohibit any writes to this extent. */
2816 pExtent->uAppendPosition = 0;
2817 }
2818
2819 if (RT_SUCCESS(rc))
2820 {
2821 pExtent->uVersion = RT_LE2H_U32(Header.version);
2822 pExtent->enmType = VMDKETYPE_HOSTED_SPARSE; /* Just dummy value, changed later. */
2823 pExtent->cSectors = RT_LE2H_U64(Header.capacity);
2824 pExtent->cSectorsPerGrain = RT_LE2H_U64(Header.grainSize);
2825 pExtent->uDescriptorSector = RT_LE2H_U64(Header.descriptorOffset);
2826 pExtent->cDescriptorSectors = RT_LE2H_U64(Header.descriptorSize);
2827 pExtent->cGTEntries = RT_LE2H_U32(Header.numGTEsPerGT);
2828 pExtent->cOverheadSectors = RT_LE2H_U64(Header.overHead);
2829 pExtent->fUncleanShutdown = !!Header.uncleanShutdown;
2830 pExtent->uCompression = RT_LE2H_U16(Header.compressAlgorithm);
2831 if (RT_LE2H_U32(Header.flags) & RT_BIT(1))
2832 {
2833 pExtent->uSectorRGD = RT_LE2H_U64(Header.rgdOffset);
2834 pExtent->uSectorGD = RT_LE2H_U64(Header.gdOffset);
2835 }
2836 else
2837 {
2838 pExtent->uSectorGD = RT_LE2H_U64(Header.gdOffset);
2839 pExtent->uSectorRGD = 0;
2840 }
2841
2842 if (pExtent->uDescriptorSector && !pExtent->cDescriptorSectors)
2843 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2844 N_("VMDK: inconsistent embedded descriptor config in '%s'"), pExtent->pszFullname);
2845
2846 if ( RT_SUCCESS(rc)
2847 && ( pExtent->uSectorGD == VMDK_GD_AT_END
2848 || pExtent->uSectorRGD == VMDK_GD_AT_END)
2849 && ( !(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2850 || !(pImage->uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL)))
2851 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2852 N_("VMDK: cannot resolve grain directory offset in '%s'"), pExtent->pszFullname);
2853
2854 if (RT_SUCCESS(rc))
2855 {
2856 uint64_t cSectorsPerGDE = pExtent->cGTEntries * pExtent->cSectorsPerGrain;
2857 if (!cSectorsPerGDE || cSectorsPerGDE > UINT32_MAX)
2858 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2859 N_("VMDK: incorrect grain directory size in '%s'"), pExtent->pszFullname);
2860 else
2861 {
2862 pExtent->cSectorsPerGDE = cSectorsPerGDE;
2863 pExtent->cGDEntries = (pExtent->cSectors + cSectorsPerGDE - 1) / cSectorsPerGDE;
2864
2865 /* Fix up the number of descriptor sectors, as some flat images have
2866 * really just one, and this causes failures when inserting the UUID
2867 * values and other extra information. */
2868 if (pExtent->cDescriptorSectors != 0 && pExtent->cDescriptorSectors < 4)
2869 {
2870 /* Do it the easy way - just fix it for flat images which have no
2871 * other complicated metadata which needs space too. */
2872 if ( pExtent->uDescriptorSector + 4 < pExtent->cOverheadSectors
2873 && pExtent->cGTEntries * pExtent->cGDEntries == 0)
2874 pExtent->cDescriptorSectors = 4;
2875 }
2876 }
2877 }
2878 }
2879 }
2880 }
2881 }
2882 else
2883 {
2884 vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error reading extent header in '%s'"), pExtent->pszFullname);
2885 rc = VERR_VD_VMDK_INVALID_HEADER;
2886 }
2887
2888 if (RT_FAILURE(rc))
2889 vmdkFreeExtentData(pImage, pExtent, false);
2890
2891 return rc;
2892}
2893
2894/**
2895 * Internal: read additional metadata belonging to an extent. For those
2896 * extents which have no additional metadata just verify the information.
2897 */
2898static int vmdkReadMetaExtent(PVMDKIMAGE pImage, PVMDKEXTENT pExtent)
2899{
2900 int rc = VINF_SUCCESS;
2901
2902/* disabled the check as there are too many truncated vmdk images out there */
2903#ifdef VBOX_WITH_VMDK_STRICT_SIZE_CHECK
2904 uint64_t cbExtentSize;
2905 /* The image must be a multiple of a sector in size and contain the data
2906 * area (flat images only). If not, it means the image is at least
2907 * truncated, or even seriously garbled. */
2908 rc = vdIfIoIntFileGetSize(pImage->pIfIo, pExtent->pFile->pStorage, &cbExtentSize);
2909 if (RT_FAILURE(rc))
2910 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error getting size in '%s'"), pExtent->pszFullname);
2911 else if ( cbExtentSize != RT_ALIGN_64(cbExtentSize, 512)
2912 && (pExtent->enmType != VMDKETYPE_FLAT || pExtent->cNominalSectors + pExtent->uSectorOffset > VMDK_BYTE2SECTOR(cbExtentSize)))
2913 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2914 N_("VMDK: file size is not a multiple of 512 in '%s', file is truncated or otherwise garbled"), pExtent->pszFullname);
2915#endif /* VBOX_WITH_VMDK_STRICT_SIZE_CHECK */
2916 if ( RT_SUCCESS(rc)
2917 && pExtent->enmType == VMDKETYPE_HOSTED_SPARSE)
2918 {
2919 /* The spec says that this must be a power of two and greater than 8,
2920 * but probably they meant not less than 8. */
2921 if ( (pExtent->cSectorsPerGrain & (pExtent->cSectorsPerGrain - 1))
2922 || pExtent->cSectorsPerGrain < 8)
2923 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2924 N_("VMDK: invalid extent grain size %u in '%s'"), pExtent->cSectorsPerGrain, pExtent->pszFullname);
2925 else
2926 {
2927 /* This code requires that a grain table must hold a power of two multiple
2928 * of the number of entries per GT cache entry. */
2929 if ( (pExtent->cGTEntries & (pExtent->cGTEntries - 1))
2930 || pExtent->cGTEntries < VMDK_GT_CACHELINE_SIZE)
2931 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS,
2932 N_("VMDK: grain table cache size problem in '%s'"), pExtent->pszFullname);
2933 else
2934 {
2935 rc = vmdkAllocStreamBuffers(pImage, pExtent);
2936 if (RT_SUCCESS(rc))
2937 {
2938 /* Prohibit any writes to this streamOptimized extent. */
2939 if (pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED)
2940 pExtent->uAppendPosition = 0;
2941
2942 if ( !(pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED)
2943 || !(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2944 || !(pImage->uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL))
2945 rc = vmdkReadGrainDirectory(pImage, pExtent);
2946 else
2947 {
2948 pExtent->uGrainSectorAbs = pExtent->cOverheadSectors;
2949 pExtent->cbGrainStreamRead = 0;
2950 }
2951 }
2952 }
2953 }
2954 }
2955
2956 if (RT_FAILURE(rc))
2957 vmdkFreeExtentData(pImage, pExtent, false);
2958
2959 return rc;
2960}
2961
2962/**
2963 * Internal: write/update the metadata for a sparse extent.
2964 */
2965static int vmdkWriteMetaSparseExtent(PVMDKIMAGE pImage, PVMDKEXTENT pExtent,
2966 uint64_t uOffset, PVDIOCTX pIoCtx)
2967{
2968 SparseExtentHeader Header;
2969
2970 memset(&Header, '\0', sizeof(Header));
2971 Header.magicNumber = RT_H2LE_U32(VMDK_SPARSE_MAGICNUMBER);
2972 Header.version = RT_H2LE_U32(pExtent->uVersion);
2973 Header.flags = RT_H2LE_U32(RT_BIT(0));
2974 if (pExtent->pRGD)
2975 Header.flags |= RT_H2LE_U32(RT_BIT(1));
2976 if (pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED)
2977 Header.flags |= RT_H2LE_U32(RT_BIT(16) | RT_BIT(17));
2978 Header.capacity = RT_H2LE_U64(pExtent->cSectors);
2979 Header.grainSize = RT_H2LE_U64(pExtent->cSectorsPerGrain);
2980 Header.descriptorOffset = RT_H2LE_U64(pExtent->uDescriptorSector);
2981 Header.descriptorSize = RT_H2LE_U64(pExtent->cDescriptorSectors);
2982 Header.numGTEsPerGT = RT_H2LE_U32(pExtent->cGTEntries);
2983 if (pExtent->fFooter && uOffset == 0)
2984 {
2985 if (pExtent->pRGD)
2986 {
2987 Assert(pExtent->uSectorRGD);
2988 Header.rgdOffset = RT_H2LE_U64(VMDK_GD_AT_END);
2989 Header.gdOffset = RT_H2LE_U64(VMDK_GD_AT_END);
2990 }
2991 else
2992 Header.gdOffset = RT_H2LE_U64(VMDK_GD_AT_END);
2993 }
2994 else
2995 {
2996 if (pExtent->pRGD)
2997 {
2998 Assert(pExtent->uSectorRGD);
2999 Header.rgdOffset = RT_H2LE_U64(pExtent->uSectorRGD);
3000 Header.gdOffset = RT_H2LE_U64(pExtent->uSectorGD);
3001 }
3002 else
3003 Header.gdOffset = RT_H2LE_U64(pExtent->uSectorGD);
3004 }
3005 Header.overHead = RT_H2LE_U64(pExtent->cOverheadSectors);
3006 Header.uncleanShutdown = pExtent->fUncleanShutdown;
3007 Header.singleEndLineChar = '\n';
3008 Header.nonEndLineChar = ' ';
3009 Header.doubleEndLineChar1 = '\r';
3010 Header.doubleEndLineChar2 = '\n';
3011 Header.compressAlgorithm = RT_H2LE_U16(pExtent->uCompression);
3012
3013 int rc = vdIfIoIntFileWriteMeta(pImage->pIfIo, pExtent->pFile->pStorage,
3014 uOffset, &Header, sizeof(Header),
3015 pIoCtx, NULL, NULL);
3016 if (RT_FAILURE(rc) && (rc != VERR_VD_ASYNC_IO_IN_PROGRESS))
3017 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error writing extent header in '%s'"), pExtent->pszFullname);
3018 return rc;
3019}
3020
3021/**
3022 * Internal: free the buffers used for streamOptimized images.
3023 */
3024static void vmdkFreeStreamBuffers(PVMDKEXTENT pExtent)
3025{
3026 if (pExtent->pvCompGrain)
3027 {
3028 RTMemFree(pExtent->pvCompGrain);
3029 pExtent->pvCompGrain = NULL;
3030 }
3031 if (pExtent->pvGrain)
3032 {
3033 RTMemFree(pExtent->pvGrain);
3034 pExtent->pvGrain = NULL;
3035 }
3036}
3037
3038/**
3039 * Internal: free the memory used by the extent data structure, optionally
3040 * deleting the referenced files.
3041 *
3042 * @returns VBox status code.
3043 * @param pImage Pointer to the image instance data.
3044 * @param pExtent The extent to free.
3045 * @param fDelete Flag whether to delete the backing storage.
3046 */
3047static int vmdkFreeExtentData(PVMDKIMAGE pImage, PVMDKEXTENT pExtent,
3048 bool fDelete)
3049{
3050 int rc = VINF_SUCCESS;
3051
3052 vmdkFreeGrainDirectory(pExtent);
3053 if (pExtent->pDescData)
3054 {
3055 RTMemFree(pExtent->pDescData);
3056 pExtent->pDescData = NULL;
3057 }
3058 if (pExtent->pFile != NULL)
3059 {
3060 /* Do not delete raw extents, these have full and base names equal. */
3061 rc = vmdkFileClose(pImage, &pExtent->pFile,
3062 fDelete
3063 && pExtent->pszFullname
3064 && pExtent->pszBasename
3065 && strcmp(pExtent->pszFullname, pExtent->pszBasename));
3066 }
3067 if (pExtent->pszBasename)
3068 {
3069 RTMemTmpFree((void *)pExtent->pszBasename);
3070 pExtent->pszBasename = NULL;
3071 }
3072 if (pExtent->pszFullname)
3073 {
3074 RTStrFree((char *)(void *)pExtent->pszFullname);
3075 pExtent->pszFullname = NULL;
3076 }
3077 vmdkFreeStreamBuffers(pExtent);
3078
3079 return rc;
3080}
3081
3082/**
3083 * Internal: allocate grain table cache if necessary for this image.
3084 */
3085static int vmdkAllocateGrainTableCache(PVMDKIMAGE pImage)
3086{
3087 PVMDKEXTENT pExtent;
3088
3089 /* Allocate grain table cache if any sparse extent is present. */
3090 for (unsigned i = 0; i < pImage->cExtents; i++)
3091 {
3092 pExtent = &pImage->pExtents[i];
3093 if (pExtent->enmType == VMDKETYPE_HOSTED_SPARSE)
3094 {
3095 /* Allocate grain table cache. */
3096 pImage->pGTCache = (PVMDKGTCACHE)RTMemAllocZ(sizeof(VMDKGTCACHE));
3097 if (!pImage->pGTCache)
3098 return VERR_NO_MEMORY;
3099 for (unsigned j = 0; j < VMDK_GT_CACHE_SIZE; j++)
3100 {
3101 PVMDKGTCACHEENTRY pGCE = &pImage->pGTCache->aGTCache[j];
3102 pGCE->uExtent = UINT32_MAX;
3103 }
3104 pImage->pGTCache->cEntries = VMDK_GT_CACHE_SIZE;
3105 break;
3106 }
3107 }
3108
3109 return VINF_SUCCESS;
3110}
3111
3112/**
3113 * Internal: allocate the given number of extents.
3114 */
3115static int vmdkCreateExtents(PVMDKIMAGE pImage, unsigned cExtents)
3116{
3117 int rc = VINF_SUCCESS;
3118 PVMDKEXTENT pExtents = (PVMDKEXTENT)RTMemAllocZ(cExtents * sizeof(VMDKEXTENT));
3119 if (pExtents)
3120 {
3121 for (unsigned i = 0; i < cExtents; i++)
3122 {
3123 pExtents[i].pFile = NULL;
3124 pExtents[i].pszBasename = NULL;
3125 pExtents[i].pszFullname = NULL;
3126 pExtents[i].pGD = NULL;
3127 pExtents[i].pRGD = NULL;
3128 pExtents[i].pDescData = NULL;
3129 pExtents[i].uVersion = 1;
3130 pExtents[i].uCompression = VMDK_COMPRESSION_NONE;
3131 pExtents[i].uExtent = i;
3132 pExtents[i].pImage = pImage;
3133 }
3134 pImage->pExtents = pExtents;
3135 pImage->cExtents = cExtents;
3136 }
3137 else
3138 rc = VERR_NO_MEMORY;
3139
3140 return rc;
3141}
3142
3143/**
3144 * Internal: Create an additional file backed extent in split images.
3145 * Supports split sparse and flat images.
3146 *
3147 * @returns VBox status code.
3148 * @param pImage VMDK image instance.
3149 * @param cbSize Desiried size in bytes of new extent.
3150 */
3151static int vmdkAddFileBackedExtent(PVMDKIMAGE pImage, uint64_t cbSize)
3152{
3153 int rc = VINF_SUCCESS;
3154 unsigned uImageFlags = pImage->uImageFlags;
3155
3156 /* Check for unsupported image type. */
3157 if ((uImageFlags & VD_VMDK_IMAGE_FLAGS_ESX)
3158 || (uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED)
3159 || (uImageFlags & VD_VMDK_IMAGE_FLAGS_RAWDISK))
3160 {
3161 return VERR_NOT_SUPPORTED;
3162 }
3163
3164 /* Allocate array of extents and copy existing extents to it. */
3165 PVMDKEXTENT pNewExtents = (PVMDKEXTENT)RTMemAllocZ((pImage->cExtents + 1) * sizeof(VMDKEXTENT));
3166 if (!pNewExtents)
3167 {
3168 return VERR_NO_MEMORY;
3169 }
3170
3171 memcpy(pNewExtents, pImage->pExtents, pImage->cExtents * sizeof(VMDKEXTENT));
3172
3173 /* Locate newly created extent and populate default metadata. */
3174 PVMDKEXTENT pExtent = &pNewExtents[pImage->cExtents];
3175
3176 pExtent->pFile = NULL;
3177 pExtent->pszBasename = NULL;
3178 pExtent->pszFullname = NULL;
3179 pExtent->pGD = NULL;
3180 pExtent->pRGD = NULL;
3181 pExtent->pDescData = NULL;
3182 pExtent->uVersion = 1;
3183 pExtent->uCompression = VMDK_COMPRESSION_NONE;
3184 pExtent->uExtent = pImage->cExtents;
3185 pExtent->pImage = pImage;
3186 pExtent->cNominalSectors = VMDK_BYTE2SECTOR(cbSize);
3187 pExtent->enmAccess = VMDKACCESS_READWRITE;
3188 pExtent->uSectorOffset = 0;
3189 pExtent->fMetaDirty = true;
3190
3191 /* Apply image type specific meta data. */
3192 if (uImageFlags & VD_IMAGE_FLAGS_FIXED)
3193 {
3194 pExtent->enmType = VMDKETYPE_FLAT;
3195 }
3196 else
3197 {
3198 uint64_t cSectorsPerGDE, cSectorsPerGD;
3199 pExtent->enmType = VMDKETYPE_HOSTED_SPARSE;
3200 pExtent->cSectors = VMDK_BYTE2SECTOR(RT_ALIGN_64(cbSize, _64K));
3201 pExtent->cSectorsPerGrain = VMDK_BYTE2SECTOR(_64K);
3202 pExtent->cGTEntries = 512;
3203 cSectorsPerGDE = pExtent->cGTEntries * pExtent->cSectorsPerGrain;
3204 pExtent->cSectorsPerGDE = cSectorsPerGDE;
3205 pExtent->cGDEntries = (pExtent->cSectors + cSectorsPerGDE - 1) / cSectorsPerGDE;
3206 cSectorsPerGD = (pExtent->cGDEntries + (512 / sizeof(uint32_t) - 1)) / (512 / sizeof(uint32_t));
3207 }
3208
3209 /* Allocate and set file name for extent. */
3210 char *pszBasenameSubstr = RTPathFilename(pImage->pszFilename);
3211 AssertPtr(pszBasenameSubstr);
3212
3213 char *pszBasenameSuff = RTPathSuffix(pszBasenameSubstr);
3214 char *pszBasenameBase = RTStrDup(pszBasenameSubstr);
3215 RTPathStripSuffix(pszBasenameBase);
3216 char *pszTmp;
3217 size_t cbTmp;
3218
3219 if (pImage->uImageFlags & VD_IMAGE_FLAGS_FIXED)
3220 RTStrAPrintf(&pszTmp, "%s-f%03d%s", pszBasenameBase,
3221 pExtent->uExtent + 1, pszBasenameSuff);
3222 else
3223 RTStrAPrintf(&pszTmp, "%s-s%03d%s", pszBasenameBase, pExtent->uExtent + 1,
3224 pszBasenameSuff);
3225
3226 RTStrFree(pszBasenameBase);
3227 if (!pszTmp)
3228 return VERR_NO_STR_MEMORY;
3229 cbTmp = strlen(pszTmp) + 1;
3230 char *pszBasename = (char *)RTMemTmpAlloc(cbTmp);
3231 if (!pszBasename)
3232 {
3233 RTStrFree(pszTmp);
3234 return VERR_NO_MEMORY;
3235 }
3236
3237 memcpy(pszBasename, pszTmp, cbTmp);
3238 RTStrFree(pszTmp);
3239
3240 pExtent->pszBasename = pszBasename;
3241
3242 char *pszBasedirectory = RTStrDup(pImage->pszFilename);
3243 if (!pszBasedirectory)
3244 return VERR_NO_STR_MEMORY;
3245 RTPathStripFilename(pszBasedirectory);
3246 char *pszFullname = RTPathJoinA(pszBasedirectory, pExtent->pszBasename);
3247 RTStrFree(pszBasedirectory);
3248 if (!pszFullname)
3249 return VERR_NO_STR_MEMORY;
3250 pExtent->pszFullname = pszFullname;
3251
3252 /* Create file for extent. */
3253 rc = vmdkFileOpen(pImage, &pExtent->pFile, pExtent->pszBasename, pExtent->pszFullname,
3254 VDOpenFlagsToFileOpenFlags(pImage->uOpenFlags,
3255 true /* fCreate */));
3256 if (RT_FAILURE(rc))
3257 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: could not create new file '%s'"), pExtent->pszFullname);
3258
3259 if (uImageFlags & VD_IMAGE_FLAGS_FIXED)
3260 {
3261 /* For flat images: Pre allocate file space. */
3262 rc = vdIfIoIntFileSetAllocationSize(pImage->pIfIo, pExtent->pFile->pStorage, cbSize,
3263 0 /* fFlags */, NULL, 0, 0);
3264 if (RT_FAILURE(rc))
3265 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: could not set size of new file '%s'"), pExtent->pszFullname);
3266 }
3267 else
3268 {
3269 /* For sparse images: Allocate new grain directories/tables. */
3270 /* fPreAlloc should never be false because VMware can't use such images. */
3271 rc = vmdkCreateGrainDirectory(pImage, pExtent,
3272 RT_MAX( pExtent->uDescriptorSector
3273 + pExtent->cDescriptorSectors,
3274 1),
3275 true /* fPreAlloc */);
3276 if (RT_FAILURE(rc))
3277 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: could not create new grain directory in '%s'"), pExtent->pszFullname);
3278 }
3279
3280 /* Insert new extent into descriptor file. */
3281 rc = vmdkDescExtInsert(pImage, &pImage->Descriptor, pExtent->enmAccess,
3282 pExtent->cNominalSectors, pExtent->enmType,
3283 pExtent->pszBasename, pExtent->uSectorOffset);
3284 if (RT_FAILURE(rc))
3285 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: could not insert the extent list into descriptor in '%s'"), pImage->pszFilename);
3286
3287 pImage->pExtents = pNewExtents;
3288 pImage->cExtents++;
3289
3290 return rc;
3291}
3292
3293/**
3294 * Reads and processes the descriptor embedded in sparse images.
3295 *
3296 * @returns VBox status code.
3297 * @param pImage VMDK image instance.
3298 * @param pFile The sparse file handle.
3299 */
3300static int vmdkDescriptorReadSparse(PVMDKIMAGE pImage, PVMDKFILE pFile)
3301{
3302 /* It's a hosted single-extent image. */
3303 int rc = vmdkCreateExtents(pImage, 1);
3304 if (RT_SUCCESS(rc))
3305 {
3306 /* The opened file is passed to the extent. No separate descriptor
3307 * file, so no need to keep anything open for the image. */
3308 PVMDKEXTENT pExtent = &pImage->pExtents[0];
3309 pExtent->pFile = pFile;
3310 pImage->pFile = NULL;
3311 pExtent->pszFullname = RTPathAbsDup(pImage->pszFilename);
3312 if (RT_LIKELY(pExtent->pszFullname))
3313 {
3314 /* As we're dealing with a monolithic image here, there must
3315 * be a descriptor embedded in the image file. */
3316 rc = vmdkReadBinaryMetaExtent(pImage, pExtent, true /* fMagicAlreadyRead */);
3317 if ( RT_SUCCESS(rc)
3318 && pExtent->uDescriptorSector
3319 && pExtent->cDescriptorSectors)
3320 {
3321 /* HACK: extend the descriptor if it is unusually small and it fits in
3322 * the unused space after the image header. Allows opening VMDK files
3323 * with extremely small descriptor in read/write mode.
3324 *
3325 * The previous version introduced a possible regression for VMDK stream
3326 * optimized images from VMware which tend to have only a single sector sized
3327 * descriptor. Increasing the descriptor size resulted in adding the various uuid
3328 * entries required to make it work with VBox but for stream optimized images
3329 * the updated binary header wasn't written to the disk creating a mismatch
3330 * between advertised and real descriptor size.
3331 *
3332 * The descriptor size will be increased even if opened readonly now if there
3333 * enough room but the new value will not be written back to the image.
3334 */
3335 if ( pExtent->cDescriptorSectors < 3
3336 && (int64_t)pExtent->uSectorGD - pExtent->uDescriptorSector >= 4
3337 && (!pExtent->uSectorRGD || (int64_t)pExtent->uSectorRGD - pExtent->uDescriptorSector >= 4))
3338 {
3339 uint64_t cDescriptorSectorsOld = pExtent->cDescriptorSectors;
3340
3341 pExtent->cDescriptorSectors = 4;
3342 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
3343 {
3344 /*
3345 * Update the on disk number now to make sure we don't introduce inconsistencies
3346 * in case of stream optimized images from VMware where the descriptor is just
3347 * one sector big (the binary header is not written to disk for complete
3348 * stream optimized images in vmdkFlushImage()).
3349 */
3350 uint64_t u64DescSizeNew = RT_H2LE_U64(pExtent->cDescriptorSectors);
3351 rc = vdIfIoIntFileWriteSync(pImage->pIfIo, pFile->pStorage,
3352 RT_UOFFSETOF(SparseExtentHeader, descriptorSize),
3353 &u64DescSizeNew, sizeof(u64DescSizeNew));
3354 if (RT_FAILURE(rc))
3355 {
3356 LogFlowFunc(("Increasing the descriptor size failed with %Rrc\n", rc));
3357 /* Restore the old size and carry on. */
3358 pExtent->cDescriptorSectors = cDescriptorSectorsOld;
3359 }
3360 }
3361 }
3362 /* Read the descriptor from the extent. */
3363 pExtent->pDescData = (char *)RTMemAllocZ(VMDK_SECTOR2BYTE(pExtent->cDescriptorSectors));
3364 if (RT_LIKELY(pExtent->pDescData))
3365 {
3366 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pExtent->pFile->pStorage,
3367 VMDK_SECTOR2BYTE(pExtent->uDescriptorSector),
3368 pExtent->pDescData,
3369 VMDK_SECTOR2BYTE(pExtent->cDescriptorSectors));
3370 if (RT_SUCCESS(rc))
3371 {
3372 rc = vmdkParseDescriptor(pImage, pExtent->pDescData,
3373 VMDK_SECTOR2BYTE(pExtent->cDescriptorSectors));
3374 if ( RT_SUCCESS(rc)
3375 && ( !(pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED)
3376 || !(pImage->uOpenFlags & VD_OPEN_FLAGS_ASYNC_IO)))
3377 {
3378 rc = vmdkReadMetaExtent(pImage, pExtent);
3379 if (RT_SUCCESS(rc))
3380 {
3381 /* Mark the extent as unclean if opened in read-write mode. */
3382 if ( !(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
3383 && !(pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED))
3384 {
3385 pExtent->fUncleanShutdown = true;
3386 pExtent->fMetaDirty = true;
3387 }
3388 }
3389 }
3390 else if (RT_SUCCESS(rc))
3391 rc = VERR_NOT_SUPPORTED;
3392 }
3393 else
3394 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: read error for descriptor in '%s'"), pExtent->pszFullname);
3395 }
3396 else
3397 rc = VERR_NO_MEMORY;
3398 }
3399 else if (RT_SUCCESS(rc))
3400 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: monolithic image without descriptor in '%s'"), pImage->pszFilename);
3401 }
3402 else
3403 rc = VERR_NO_MEMORY;
3404 }
3405
3406 return rc;
3407}
3408
3409/**
3410 * Reads the descriptor from a pure text file.
3411 *
3412 * @returns VBox status code.
3413 * @param pImage VMDK image instance.
3414 * @param pFile The descriptor file handle.
3415 */
3416static int vmdkDescriptorReadAscii(PVMDKIMAGE pImage, PVMDKFILE pFile)
3417{
3418 /* Allocate at least 10K, and make sure that there is 5K free space
3419 * in case new entries need to be added to the descriptor. Never
3420 * allocate more than 128K, because that's no valid descriptor file
3421 * and will result in the correct "truncated read" error handling. */
3422 uint64_t cbFileSize;
3423 int rc = vdIfIoIntFileGetSize(pImage->pIfIo, pFile->pStorage, &cbFileSize);
3424 if ( RT_SUCCESS(rc)
3425 && cbFileSize >= 50)
3426 {
3427 uint64_t cbSize = cbFileSize;
3428 if (cbSize % VMDK_SECTOR2BYTE(10))
3429 cbSize += VMDK_SECTOR2BYTE(20) - cbSize % VMDK_SECTOR2BYTE(10);
3430 else
3431 cbSize += VMDK_SECTOR2BYTE(10);
3432 cbSize = RT_MIN(cbSize, _128K);
3433 pImage->cbDescAlloc = RT_MAX(VMDK_SECTOR2BYTE(20), cbSize);
3434 pImage->pDescData = (char *)RTMemAllocZ(pImage->cbDescAlloc);
3435 if (RT_LIKELY(pImage->pDescData))
3436 {
3437 rc = vdIfIoIntFileReadSync(pImage->pIfIo, pFile->pStorage, 0, pImage->pDescData,
3438 RT_MIN(pImage->cbDescAlloc, cbFileSize));
3439 if (RT_SUCCESS(rc))
3440 {
3441#if 0 /** @todo Revisit */
3442 cbRead += sizeof(u32Magic);
3443 if (cbRead == pImage->cbDescAlloc)
3444 {
3445 /* Likely the read is truncated. Better fail a bit too early
3446 * (normally the descriptor is much smaller than our buffer). */
3447 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: cannot read descriptor in '%s'"), pImage->pszFilename);
3448 goto out;
3449 }
3450#endif
3451 rc = vmdkParseDescriptor(pImage, pImage->pDescData,
3452 pImage->cbDescAlloc);
3453 if (RT_SUCCESS(rc))
3454 {
3455 for (unsigned i = 0; i < pImage->cExtents && RT_SUCCESS(rc); i++)
3456 {
3457 PVMDKEXTENT pExtent = &pImage->pExtents[i];
3458 if (pExtent->pszBasename)
3459 {
3460 /* Hack to figure out whether the specified name in the
3461 * extent descriptor is absolute. Doesn't always work, but
3462 * should be good enough for now. */
3463 char *pszFullname;
3464 /** @todo implement proper path absolute check. */
3465 if (pExtent->pszBasename[0] == RTPATH_SLASH)
3466 {
3467 pszFullname = RTStrDup(pExtent->pszBasename);
3468 if (!pszFullname)
3469 {
3470 rc = VERR_NO_MEMORY;
3471 break;
3472 }
3473 }
3474 else
3475 {
3476 char *pszDirname = RTStrDup(pImage->pszFilename);
3477 if (!pszDirname)
3478 {
3479 rc = VERR_NO_MEMORY;
3480 break;
3481 }
3482 RTPathStripFilename(pszDirname);
3483 pszFullname = RTPathJoinA(pszDirname, pExtent->pszBasename);
3484 RTStrFree(pszDirname);
3485 if (!pszFullname)
3486 {
3487 rc = VERR_NO_STR_MEMORY;
3488 break;
3489 }
3490 }
3491 pExtent->pszFullname = pszFullname;
3492 }
3493 else
3494 pExtent->pszFullname = NULL;
3495
3496 unsigned uOpenFlags = pImage->uOpenFlags | ((pExtent->enmAccess == VMDKACCESS_READONLY) ? VD_OPEN_FLAGS_READONLY : 0);
3497 switch (pExtent->enmType)
3498 {
3499 case VMDKETYPE_HOSTED_SPARSE:
3500 rc = vmdkFileOpen(pImage, &pExtent->pFile, pExtent->pszBasename, pExtent->pszFullname,
3501 VDOpenFlagsToFileOpenFlags(uOpenFlags, false /* fCreate */));
3502 if (RT_FAILURE(rc))
3503 {
3504 /* Do NOT signal an appropriate error here, as the VD
3505 * layer has the choice of retrying the open if it
3506 * failed. */
3507 break;
3508 }
3509 rc = vmdkReadBinaryMetaExtent(pImage, pExtent,
3510 false /* fMagicAlreadyRead */);
3511 if (RT_FAILURE(rc))
3512 break;
3513 rc = vmdkReadMetaExtent(pImage, pExtent);
3514 if (RT_FAILURE(rc))
3515 break;
3516
3517 /* Mark extent as unclean if opened in read-write mode. */
3518 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
3519 {
3520 pExtent->fUncleanShutdown = true;
3521 pExtent->fMetaDirty = true;
3522 }
3523 break;
3524 case VMDKETYPE_VMFS:
3525 case VMDKETYPE_FLAT:
3526 rc = vmdkFileOpen(pImage, &pExtent->pFile, pExtent->pszBasename, pExtent->pszFullname,
3527 VDOpenFlagsToFileOpenFlags(uOpenFlags, false /* fCreate */));
3528 if (RT_FAILURE(rc))
3529 {
3530 /* Do NOT signal an appropriate error here, as the VD
3531 * layer has the choice of retrying the open if it
3532 * failed. */
3533 break;
3534 }
3535 break;
3536 case VMDKETYPE_ZERO:
3537 /* Nothing to do. */
3538 break;
3539 default:
3540 AssertMsgFailed(("unknown vmdk extent type %d\n", pExtent->enmType));
3541 }
3542 }
3543 }
3544 }
3545 else
3546 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: read error for descriptor in '%s'"), pImage->pszFilename);
3547 }
3548 else
3549 rc = VERR_NO_MEMORY;
3550 }
3551 else if (RT_SUCCESS(rc))
3552 rc = vdIfError(pImage->pIfError, VERR_VD_VMDK_INVALID_HEADER, RT_SRC_POS, N_("VMDK: descriptor in '%s' is too short"), pImage->pszFilename);
3553
3554 return rc;
3555}
3556
3557/**
3558 * Read and process the descriptor based on the image type.
3559 *
3560 * @returns VBox status code.
3561 * @param pImage VMDK image instance.
3562 * @param pFile VMDK file handle.
3563 */
3564static int vmdkDescriptorRead(PVMDKIMAGE pImage, PVMDKFILE pFile)
3565{
3566 uint32_t u32Magic;
3567
3568 /* Read magic (if present). */
3569 int rc = vdIfIoIntFileReadSync(pImage->pIfIo, pFile->pStorage, 0,
3570 &u32Magic, sizeof(u32Magic));
3571 if (RT_SUCCESS(rc))
3572 {
3573 /* Handle the file according to its magic number. */
3574 if (RT_LE2H_U32(u32Magic) == VMDK_SPARSE_MAGICNUMBER)
3575 rc = vmdkDescriptorReadSparse(pImage, pFile);
3576 else
3577 rc = vmdkDescriptorReadAscii(pImage, pFile);
3578 }
3579 else
3580 {
3581 vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: error reading the magic number in '%s'"), pImage->pszFilename);
3582 rc = VERR_VD_VMDK_INVALID_HEADER;
3583 }
3584
3585 return rc;
3586}
3587
3588/**
3589 * Internal: Open an image, constructing all necessary data structures.
3590 */
3591static int vmdkOpenImage(PVMDKIMAGE pImage, unsigned uOpenFlags)
3592{
3593 pImage->uOpenFlags = uOpenFlags;
3594 pImage->pIfError = VDIfErrorGet(pImage->pVDIfsDisk);
3595 pImage->pIfIo = VDIfIoIntGet(pImage->pVDIfsImage);
3596 AssertPtrReturn(pImage->pIfIo, VERR_INVALID_PARAMETER);
3597
3598 /*
3599 * Open the image.
3600 * We don't have to check for asynchronous access because
3601 * we only support raw access and the opened file is a description
3602 * file were no data is stored.
3603 */
3604 PVMDKFILE pFile;
3605 int rc = vmdkFileOpen(pImage, &pFile, NULL, pImage->pszFilename,
3606 VDOpenFlagsToFileOpenFlags(uOpenFlags, false /* fCreate */));
3607 if (RT_SUCCESS(rc))
3608 {
3609 pImage->pFile = pFile;
3610
3611 rc = vmdkDescriptorRead(pImage, pFile);
3612 if (RT_SUCCESS(rc))
3613 {
3614 /* Determine PCHS geometry if not set. */
3615 if (pImage->PCHSGeometry.cCylinders == 0)
3616 {
3617 uint64_t cCylinders = VMDK_BYTE2SECTOR(pImage->cbSize)
3618 / pImage->PCHSGeometry.cHeads
3619 / pImage->PCHSGeometry.cSectors;
3620 pImage->PCHSGeometry.cCylinders = (unsigned)RT_MIN(cCylinders, 16383);
3621 if ( !(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
3622 && !(pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED))
3623 {
3624 rc = vmdkDescSetPCHSGeometry(pImage, &pImage->PCHSGeometry);
3625 AssertRC(rc);
3626 }
3627 }
3628
3629 /* Update the image metadata now in case has changed. */
3630 rc = vmdkFlushImage(pImage, NULL);
3631 if (RT_SUCCESS(rc))
3632 {
3633 /* Figure out a few per-image constants from the extents. */
3634 pImage->cbSize = 0;
3635 for (unsigned i = 0; i < pImage->cExtents; i++)
3636 {
3637 PVMDKEXTENT pExtent = &pImage->pExtents[i];
3638 if (pExtent->enmType == VMDKETYPE_HOSTED_SPARSE)
3639 {
3640 /* Here used to be a check whether the nominal size of an extent
3641 * is a multiple of the grain size. The spec says that this is
3642 * always the case, but unfortunately some files out there in the
3643 * wild violate the spec (e.g. ReactOS 0.3.1). */
3644 }
3645 else if ( pExtent->enmType == VMDKETYPE_FLAT
3646 || pExtent->enmType == VMDKETYPE_ZERO)
3647 pImage->uImageFlags |= VD_IMAGE_FLAGS_FIXED;
3648
3649 pImage->cbSize += VMDK_SECTOR2BYTE(pExtent->cNominalSectors);
3650 }
3651
3652 if ( !(pImage->uImageFlags & VD_VMDK_IMAGE_FLAGS_STREAM_OPTIMIZED)
3653 || !(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
3654 || !(pImage->uOpenFlags & VD_OPEN_FLAGS_SEQUENTIAL))
3655 rc = vmdkAllocateGrainTableCache(pImage);
3656 }
3657 }
3658 }
3659 /* else: Do NOT signal an appropriate error here, as the VD layer has the
3660 * choice of retrying the open if it failed. */
3661
3662 if (RT_SUCCESS(rc))
3663 {
3664 PVDREGIONDESC pRegion = &pImage->RegionList.aRegions[0];
3665 pImage->RegionList.fFlags = 0;
3666 pImage->RegionList.cRegions = 1;
3667
3668 pRegion->offRegion = 0; /* Disk start. */
3669 pRegion->cbBlock = 512;
3670 pRegion->enmDataForm = VDREGIONDATAFORM_RAW;
3671 pRegion->enmMetadataForm = VDREGIONMETADATAFORM_NONE;
3672 pRegion->cbData = 512;
3673 pRegion->cbMetadata = 0;
3674 pRegion->cRegionBlocksOrBytes = pImage->cbSize;
3675 }
3676 else
3677 vmdkFreeImage(pImage, false, false /*fFlush*/); /* Don't try to flush anything if opening failed. */
3678 return rc;
3679}
3680
3681/**
3682 * Frees a raw descriptor.
3683 * @internal
3684 */
3685static int vmdkRawDescFree(PVDISKRAW pRawDesc)
3686{
3687 if (!pRawDesc)
3688 return VINF_SUCCESS;
3689
3690 RTStrFree(pRawDesc->pszRawDisk);
3691 pRawDesc->pszRawDisk = NULL;
3692
3693 /* Partitions: */
3694 for (unsigned i = 0; i < pRawDesc->cPartDescs; i++)
3695 {
3696 RTStrFree(pRawDesc->pPartDescs[i].pszRawDevice);
3697 pRawDesc->pPartDescs[i].pszRawDevice = NULL;
3698
3699 RTMemFree(pRawDesc->pPartDescs[i].pvPartitionData);
3700 pRawDesc->pPartDescs[i].pvPartitionData = NULL;
3701 }
3702
3703 RTMemFree(pRawDesc->pPartDescs);
3704 pRawDesc->pPartDescs = NULL;
3705
3706 RTMemFree(pRawDesc);
3707 return VINF_SUCCESS;
3708}
3709
3710/**
3711 * Helper that grows the raw partition descriptor table by @a cToAdd entries,
3712 * returning the pointer to the first new entry.
3713 * @internal
3714 */
3715static int vmdkRawDescAppendPartDesc(PVMDKIMAGE pImage, PVDISKRAW pRawDesc, uint32_t cToAdd, PVDISKRAWPARTDESC *ppRet)
3716{
3717 uint32_t const cOld = pRawDesc->cPartDescs;
3718 uint32_t const cNew = cOld + cToAdd;
3719 PVDISKRAWPARTDESC paNew = (PVDISKRAWPARTDESC)RTMemReallocZ(pRawDesc->pPartDescs,
3720 cOld * sizeof(pRawDesc->pPartDescs[0]),
3721 cNew * sizeof(pRawDesc->pPartDescs[0]));
3722 if (paNew)
3723 {
3724 pRawDesc->cPartDescs = cNew;
3725 pRawDesc->pPartDescs = paNew;
3726
3727 *ppRet = &paNew[cOld];
3728 return VINF_SUCCESS;
3729 }
3730 *ppRet = NULL;
3731 return vdIfError(pImage->pIfError, VERR_NO_MEMORY, RT_SRC_POS,
3732 N_("VMDK: Image path: '%s'. Out of memory growing the partition descriptors (%u -> %u)."),
3733 pImage->pszFilename, cOld, cNew);
3734}
3735
3736/**
3737 * @callback_method_impl{FNRTSORTCMP}
3738 */
3739static DECLCALLBACK(int) vmdkRawDescPartComp(void const *pvElement1, void const *pvElement2, void *pvUser)
3740{
3741 RT_NOREF(pvUser);
3742 int64_t const iDelta = ((PVDISKRAWPARTDESC)pvElement1)->offStartInVDisk - ((PVDISKRAWPARTDESC)pvElement2)->offStartInVDisk;
3743 return iDelta < 0 ? -1 : iDelta > 0 ? 1 : 0;
3744}
3745
3746/**
3747 * Post processes the partition descriptors.
3748 *
3749 * Sorts them and check that they don't overlap.
3750 */
3751static int vmdkRawDescPostProcessPartitions(PVMDKIMAGE pImage, PVDISKRAW pRawDesc, uint64_t cbSize)
3752{
3753 /*
3754 * Sort data areas in ascending order of start.
3755 */
3756 RTSortShell(pRawDesc->pPartDescs, pRawDesc->cPartDescs, sizeof(pRawDesc->pPartDescs[0]), vmdkRawDescPartComp, NULL);
3757
3758 /*
3759 * Check that we don't have overlapping descriptors. If we do, that's an
3760 * indication that the drive is corrupt or that the RTDvm code is buggy.
3761 */
3762 VDISKRAWPARTDESC const *paPartDescs = pRawDesc->pPartDescs;
3763 for (uint32_t i = 0; i < pRawDesc->cPartDescs; i++)
3764 {
3765 uint64_t offLast = paPartDescs[i].offStartInVDisk + paPartDescs[i].cbData;
3766 if (offLast <= paPartDescs[i].offStartInVDisk)
3767 return vdIfError(pImage->pIfError, VERR_FILESYSTEM_CORRUPT /*?*/, RT_SRC_POS,
3768 N_("VMDK: Image path: '%s'. Bogus partition descriptor #%u (%#RX64 LB %#RX64%s): Wrap around or zero"),
3769 pImage->pszFilename, i, paPartDescs[i].offStartInVDisk, paPartDescs[i].cbData,
3770 paPartDescs[i].pvPartitionData ? " (data)" : "");
3771 offLast -= 1;
3772
3773 if (i + 1 < pRawDesc->cPartDescs && offLast >= paPartDescs[i + 1].offStartInVDisk)
3774 return vdIfError(pImage->pIfError, VERR_FILESYSTEM_CORRUPT /*?*/, RT_SRC_POS,
3775 N_("VMDK: Image path: '%s'. Partition descriptor #%u (%#RX64 LB %#RX64%s) overlaps with the next (%#RX64 LB %#RX64%s)"),
3776 pImage->pszFilename, i, paPartDescs[i].offStartInVDisk, paPartDescs[i].cbData,
3777 paPartDescs[i].pvPartitionData ? " (data)" : "", paPartDescs[i + 1].offStartInVDisk,
3778 paPartDescs[i + 1].cbData, paPartDescs[i + 1].pvPartitionData ? " (data)" : "");
3779 if (offLast >= cbSize)
3780 return vdIfError(pImage->pIfError, VERR_FILESYSTEM_CORRUPT /*?*/, RT_SRC_POS,
3781 N_("VMDK: Image path: '%s'. Partition descriptor #%u (%#RX64 LB %#RX64%s) goes beyond the end of the drive (%#RX64)"),
3782 pImage->pszFilename, i, paPartDescs[i].offStartInVDisk, paPartDescs[i].cbData,
3783 paPartDescs[i].pvPartitionData ? " (data)" : "", cbSize);
3784 }
3785
3786 return VINF_SUCCESS;
3787}
3788
3789
3790#ifdef RT_OS_LINUX
3791/**
3792 * Searches the dir specified in @a pszBlockDevDir for subdirectories with a
3793 * 'dev' file matching @a uDevToLocate.
3794 *
3795 * This is used both
3796 *
3797 * @returns IPRT status code, errors have been reported properly.
3798 * @param pImage For error reporting.
3799 * @param pszBlockDevDir Input: Path to the directory search under.
3800 * Output: Path to the directory containing information
3801 * for @a uDevToLocate.
3802 * @param cbBlockDevDir The size of the buffer @a pszBlockDevDir points to.
3803 * @param uDevToLocate The device number of the block device info dir to
3804 * locate.
3805 * @param pszDevToLocate For error reporting.
3806 */
3807static int vmdkFindSysBlockDevPath(PVMDKIMAGE pImage, char *pszBlockDevDir, size_t cbBlockDevDir,
3808 dev_t uDevToLocate, const char *pszDevToLocate)
3809{
3810 size_t const cchDir = RTPathEnsureTrailingSeparator(pszBlockDevDir, cbBlockDevDir);
3811 AssertReturn(cchDir > 0, VERR_BUFFER_OVERFLOW);
3812
3813 RTDIR hDir = NIL_RTDIR;
3814 int rc = RTDirOpen(&hDir, pszBlockDevDir);
3815 if (RT_SUCCESS(rc))
3816 {
3817 for (;;)
3818 {
3819 RTDIRENTRY Entry;
3820 rc = RTDirRead(hDir, &Entry, NULL);
3821 if (RT_SUCCESS(rc))
3822 {
3823 /* We're interested in directories and symlinks. */
3824 if ( Entry.enmType == RTDIRENTRYTYPE_DIRECTORY
3825 || Entry.enmType == RTDIRENTRYTYPE_SYMLINK
3826 || Entry.enmType == RTDIRENTRYTYPE_UNKNOWN)
3827 {
3828 rc = RTStrCopy(&pszBlockDevDir[cchDir], cbBlockDevDir - cchDir, Entry.szName);
3829 AssertContinue(RT_SUCCESS(rc)); /* should not happen! */
3830
3831 dev_t uThisDevNo = ~uDevToLocate;
3832 rc = RTLinuxSysFsReadDevNumFile(&uThisDevNo, "%s/dev", pszBlockDevDir);
3833 if (RT_SUCCESS(rc) && uThisDevNo == uDevToLocate)
3834 break;
3835 }
3836 }
3837 else
3838 {
3839 pszBlockDevDir[cchDir] = '\0';
3840 if (rc == VERR_NO_MORE_FILES)
3841 rc = vdIfError(pImage->pIfError, VERR_NOT_FOUND, RT_SRC_POS,
3842 N_("VMDK: Image path: '%s'. Failed to locate device corresponding to '%s' under '%s'"),
3843 pImage->pszFilename, pszDevToLocate, pszBlockDevDir);
3844 else
3845 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
3846 N_("VMDK: Image path: '%s'. RTDirRead failed enumerating '%s': %Rrc"),
3847 pImage->pszFilename, pszBlockDevDir, rc);
3848 break;
3849 }
3850 }
3851 RTDirClose(hDir);
3852 }
3853 else
3854 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
3855 N_("VMDK: Image path: '%s'. Failed to open dir '%s' for listing: %Rrc"),
3856 pImage->pszFilename, pszBlockDevDir, rc);
3857 return rc;
3858}
3859#endif /* RT_OS_LINUX */
3860
3861#ifdef RT_OS_FREEBSD
3862
3863
3864/**
3865 * Reads the config data from the provider and returns offset and size
3866 *
3867 * @return IPRT status code
3868 * @param pProvider GEOM provider representing partition
3869 * @param pcbOffset Placeholder for the offset of the partition
3870 * @param pcbSize Placeholder for the size of the partition
3871 */
3872static int vmdkReadPartitionsParamsFromProvider(gprovider *pProvider, uint64_t *pcbOffset, uint64_t *pcbSize)
3873{
3874 gconfig *pConfEntry;
3875 int rc = VERR_NOT_FOUND;
3876
3877 /*
3878 * Required parameters are located in the list containing key/value pairs.
3879 * Both key and value are in text form. Manuals tells nothing about the fact
3880 * that the both parameters should be present in the list. Thus, there are
3881 * cases when only one parameter is presented. To handle such cases we treat
3882 * absent params as zero allowing the caller decide the case is either correct
3883 * or an error.
3884 */
3885 uint64_t cbOffset = 0;
3886 uint64_t cbSize = 0;
3887 LIST_FOREACH(pConfEntry, &pProvider->lg_config, lg_config)
3888 {
3889 if (RTStrCmp(pConfEntry->lg_name, "offset") == 0)
3890 {
3891 cbOffset = RTStrToUInt64(pConfEntry->lg_val);
3892 rc = VINF_SUCCESS;
3893 }
3894 else if (RTStrCmp(pConfEntry->lg_name, "length") == 0)
3895 {
3896 cbSize = RTStrToUInt64(pConfEntry->lg_val);
3897 rc = VINF_SUCCESS;
3898 }
3899 }
3900 if (RT_SUCCESS(rc))
3901 {
3902 *pcbOffset = cbOffset;
3903 *pcbSize = cbSize;
3904 }
3905 return rc;
3906}
3907
3908
3909/**
3910 * Searches the partition specified by name and calculates its size and absolute offset.
3911 *
3912 * @return IPRT status code.
3913 * @param pParentClass Class containing pParentGeom
3914 * @param pszParentGeomName Name of the parent geom where we are looking for provider
3915 * @param pszProviderName Name of the provider we are looking for
3916 * @param pcbAbsoluteOffset Placeholder for the absolute offset of the partition, i.e. offset from the beginning of the disk
3917 * @param psbSize Placeholder for the size of the partition.
3918 */
3919static int vmdkFindPartitionParamsByName(gclass *pParentClass, const char *pszParentGeomName, const char *pszProviderName,
3920 uint64_t *pcbAbsoluteOffset, uint64_t *pcbSize)
3921{
3922 AssertReturn(pParentClass, VERR_INVALID_PARAMETER);
3923 AssertReturn(pszParentGeomName, VERR_INVALID_PARAMETER);
3924 AssertReturn(pszProviderName, VERR_INVALID_PARAMETER);
3925 AssertReturn(pcbAbsoluteOffset, VERR_INVALID_PARAMETER);
3926 AssertReturn(pcbSize, VERR_INVALID_PARAMETER);
3927
3928 ggeom *pParentGeom;
3929 int rc = VERR_NOT_FOUND;
3930 LIST_FOREACH(pParentGeom, &pParentClass->lg_geom, lg_geom)
3931 {
3932 if (RTStrCmp(pParentGeom->lg_name, pszParentGeomName) == 0)
3933 {
3934 rc = VINF_SUCCESS;
3935 break;
3936 }
3937 }
3938 if (RT_FAILURE(rc))
3939 return rc;
3940
3941 gprovider *pProvider;
3942 /*
3943 * First, go over providers without handling EBR or BSDLabel
3944 * partitions for case when looking provider is child
3945 * of the givng geom, to reduce searching time
3946 */
3947 LIST_FOREACH(pProvider, &pParentGeom->lg_provider, lg_provider)
3948 {
3949 if (RTStrCmp(pProvider->lg_name, pszProviderName) == 0)
3950 return vmdkReadPartitionsParamsFromProvider(pProvider, pcbAbsoluteOffset, pcbSize);
3951 }
3952
3953 /*
3954 * No provider found. Go over the parent geom again
3955 * and make recursions if geom represents EBR or BSDLabel.
3956 * In this case given parent geom contains only EBR or BSDLabel
3957 * partition itself and their own partitions are in the separate
3958 * geoms. Also, partition offsets are relative to geom, so
3959 * we have to add offset from child provider with parent geoms
3960 * provider
3961 */
3962
3963 LIST_FOREACH(pProvider, &pParentGeom->lg_provider, lg_provider)
3964 {
3965 uint64_t cbOffset = 0;
3966 uint64_t cbSize = 0;
3967 rc = vmdkReadPartitionsParamsFromProvider(pProvider, &cbOffset, &cbSize);
3968 if (RT_FAILURE(rc))
3969 return rc;
3970
3971 uint64_t cbProviderOffset = 0;
3972 uint64_t cbProviderSize = 0;
3973 rc = vmdkFindPartitionParamsByName(pParentClass, pProvider->lg_name, pszProviderName, &cbProviderOffset, &cbProviderSize);
3974 if (RT_SUCCESS(rc))
3975 {
3976 *pcbAbsoluteOffset = cbOffset + cbProviderOffset;
3977 *pcbSize = cbProviderSize;
3978 return rc;
3979 }
3980 }
3981
3982 return VERR_NOT_FOUND;
3983}
3984#endif
3985
3986
3987/**
3988 * Attempts to verify the raw partition path.
3989 *
3990 * We don't want to trust RTDvm and the partition device node morphing blindly.
3991 */
3992static int vmdkRawDescVerifyPartitionPath(PVMDKIMAGE pImage, PVDISKRAWPARTDESC pPartDesc, uint32_t idxPartition,
3993 const char *pszRawDrive, RTFILE hRawDrive, uint32_t cbSector, RTDVMVOLUME hVol)
3994{
3995 RT_NOREF(pImage, pPartDesc, idxPartition, pszRawDrive, hRawDrive, cbSector, hVol);
3996
3997 /*
3998 * Try open the raw partition device.
3999 */
4000 RTFILE hRawPart = NIL_RTFILE;
4001 int rc = RTFileOpen(&hRawPart, pPartDesc->pszRawDevice, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
4002 if (RT_FAILURE(rc))
4003 return vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4004 N_("VMDK: Image path: '%s'. Failed to open partition #%u on '%s' via '%s' (%Rrc)"),
4005 pImage->pszFilename, idxPartition, pszRawDrive, pPartDesc->pszRawDevice, rc);
4006
4007 /*
4008 * Compare the partition UUID if we can get it.
4009 */
4010#ifdef RT_OS_WINDOWS
4011 DWORD cbReturned;
4012
4013 /* 1. Get the device numbers for both handles, they should have the same disk. */
4014 STORAGE_DEVICE_NUMBER DevNum1;
4015 RT_ZERO(DevNum1);
4016 if (!DeviceIoControl((HANDLE)RTFileToNative(hRawDrive), IOCTL_STORAGE_GET_DEVICE_NUMBER,
4017 NULL /*pvInBuffer*/, 0 /*cbInBuffer*/, &DevNum1, sizeof(DevNum1), &cbReturned, NULL /*pOverlapped*/))
4018 rc = vdIfError(pImage->pIfError, RTErrConvertFromWin32(GetLastError()), RT_SRC_POS,
4019 N_("VMDK: Image path: '%s'. IOCTL_STORAGE_GET_DEVICE_NUMBER failed on '%s': %u"),
4020 pImage->pszFilename, pszRawDrive, GetLastError());
4021
4022 STORAGE_DEVICE_NUMBER DevNum2;
4023 RT_ZERO(DevNum2);
4024 if (!DeviceIoControl((HANDLE)RTFileToNative(hRawPart), IOCTL_STORAGE_GET_DEVICE_NUMBER,
4025 NULL /*pvInBuffer*/, 0 /*cbInBuffer*/, &DevNum2, sizeof(DevNum2), &cbReturned, NULL /*pOverlapped*/))
4026 rc = vdIfError(pImage->pIfError, RTErrConvertFromWin32(GetLastError()), RT_SRC_POS,
4027 N_("VMDK: Image path: '%s'. IOCTL_STORAGE_GET_DEVICE_NUMBER failed on '%s': %u"),
4028 pImage->pszFilename, pPartDesc->pszRawDevice, GetLastError());
4029 if ( RT_SUCCESS(rc)
4030 && ( DevNum1.DeviceNumber != DevNum2.DeviceNumber
4031 || DevNum1.DeviceType != DevNum2.DeviceType))
4032 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4033 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s' (%#x != %#x || %#x != %#x)"),
4034 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive,
4035 DevNum1.DeviceNumber, DevNum2.DeviceNumber, DevNum1.DeviceType, DevNum2.DeviceType);
4036 if (RT_SUCCESS(rc))
4037 {
4038 /* Get the partitions from the raw drive and match up with the volume info
4039 from RTDvm. The partition number is found in DevNum2. */
4040 DWORD cbNeeded = 0;
4041 if ( DeviceIoControl((HANDLE)RTFileToNative(hRawDrive), IOCTL_DISK_GET_DRIVE_LAYOUT_EX,
4042 NULL /*pvInBuffer*/, 0 /*cbInBuffer*/, NULL, 0, &cbNeeded, NULL /*pOverlapped*/)
4043 || cbNeeded < RT_UOFFSETOF_DYN(DRIVE_LAYOUT_INFORMATION_EX, PartitionEntry[1]))
4044 cbNeeded = RT_UOFFSETOF_DYN(DRIVE_LAYOUT_INFORMATION_EX, PartitionEntry[64]);
4045 cbNeeded += sizeof(PARTITION_INFORMATION_EX) * 2; /* just in case */
4046 DRIVE_LAYOUT_INFORMATION_EX *pLayout = (DRIVE_LAYOUT_INFORMATION_EX *)RTMemTmpAllocZ(cbNeeded);
4047 if (pLayout)
4048 {
4049 cbReturned = 0;
4050 if (DeviceIoControl((HANDLE)RTFileToNative(hRawDrive), IOCTL_DISK_GET_DRIVE_LAYOUT_EX,
4051 NULL /*pvInBuffer*/, 0 /*cbInBuffer*/, pLayout, cbNeeded, &cbReturned, NULL /*pOverlapped*/))
4052 {
4053 /* Find the entry with the given partition number (it's not an index, array contains empty MBR entries ++). */
4054 unsigned iEntry = 0;
4055 while ( iEntry < pLayout->PartitionCount
4056 && pLayout->PartitionEntry[iEntry].PartitionNumber != DevNum2.PartitionNumber)
4057 iEntry++;
4058 if (iEntry < pLayout->PartitionCount)
4059 {
4060 /* Compare the basics */
4061 PARTITION_INFORMATION_EX const * const pLayoutEntry = &pLayout->PartitionEntry[iEntry];
4062 if (pLayoutEntry->StartingOffset.QuadPart != (int64_t)pPartDesc->offStartInVDisk)
4063 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4064 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': StartingOffset %RU64, expected %RU64"),
4065 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive,
4066 pLayoutEntry->StartingOffset.QuadPart, pPartDesc->offStartInVDisk);
4067 else if (pLayoutEntry->PartitionLength.QuadPart != (int64_t)pPartDesc->cbData)
4068 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4069 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': PartitionLength %RU64, expected %RU64"),
4070 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive,
4071 pLayoutEntry->PartitionLength.QuadPart, pPartDesc->cbData);
4072 /** @todo We could compare the MBR type, GPT type and ID. */
4073 RT_NOREF(hVol);
4074 }
4075 else
4076 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4077 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': PartitionCount (%#x vs %#x)"),
4078 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive,
4079 DevNum2.PartitionNumber, pLayout->PartitionCount);
4080# ifndef LOG_ENABLED
4081 if (RT_FAILURE(rc))
4082# endif
4083 {
4084 LogRel(("VMDK: Windows reports %u partitions for '%s':\n", pLayout->PartitionCount, pszRawDrive));
4085 PARTITION_INFORMATION_EX const *pEntry = &pLayout->PartitionEntry[0];
4086 for (DWORD i = 0; i < pLayout->PartitionCount; i++, pEntry++)
4087 {
4088 LogRel(("VMDK: #%u/%u: %016RU64 LB %016RU64 style=%d rewrite=%d",
4089 i, pEntry->PartitionNumber, pEntry->StartingOffset.QuadPart, pEntry->PartitionLength.QuadPart,
4090 pEntry->PartitionStyle, pEntry->RewritePartition));
4091 if (pEntry->PartitionStyle == PARTITION_STYLE_MBR)
4092 LogRel((" type=%#x boot=%d rec=%d hidden=%u\n", pEntry->Mbr.PartitionType, pEntry->Mbr.BootIndicator,
4093 pEntry->Mbr.RecognizedPartition, pEntry->Mbr.HiddenSectors));
4094 else if (pEntry->PartitionStyle == PARTITION_STYLE_GPT)
4095 LogRel((" type=%RTuuid id=%RTuuid aatrib=%RX64 name=%.36ls\n", &pEntry->Gpt.PartitionType,
4096 &pEntry->Gpt.PartitionId, pEntry->Gpt.Attributes, &pEntry->Gpt.Name[0]));
4097 else
4098 LogRel(("\n"));
4099 }
4100 LogRel(("VMDK: Looked for partition #%u (%u, '%s') at %RU64 LB %RU64\n", DevNum2.PartitionNumber,
4101 idxPartition, pPartDesc->pszRawDevice, pPartDesc->offStartInVDisk, pPartDesc->cbData));
4102 }
4103 }
4104 else
4105 rc = vdIfError(pImage->pIfError, RTErrConvertFromWin32(GetLastError()), RT_SRC_POS,
4106 N_("VMDK: Image path: '%s'. IOCTL_DISK_GET_DRIVE_LAYOUT_EX failed on '%s': %u (cb %u, cbRet %u)"),
4107 pImage->pszFilename, pPartDesc->pszRawDevice, GetLastError(), cbNeeded, cbReturned);
4108 RTMemTmpFree(pLayout);
4109 }
4110 else
4111 rc = VERR_NO_TMP_MEMORY;
4112 }
4113
4114#elif defined(RT_OS_LINUX)
4115 RT_NOREF(hVol);
4116
4117 /* Stat the two devices first to get their device numbers. (We probably
4118 could make some assumptions here about the major & minor number assignments
4119 for legacy nodes, but it doesn't hold up for nvme, so we'll skip that.) */
4120 struct stat StDrive, StPart;
4121 if (fstat((int)RTFileToNative(hRawDrive), &StDrive) != 0)
4122 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4123 N_("VMDK: Image path: '%s'. fstat failed on '%s': %d"), pImage->pszFilename, pszRawDrive, errno);
4124 else if (fstat((int)RTFileToNative(hRawPart), &StPart) != 0)
4125 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4126 N_("VMDK: Image path: '%s'. fstat failed on '%s': %d"), pImage->pszFilename, pPartDesc->pszRawDevice, errno);
4127 else
4128 {
4129 /* Scan the directories immediately under /sys/block/ for one with a
4130 'dev' file matching the drive's device number: */
4131 char szSysPath[RTPATH_MAX];
4132 rc = RTLinuxConstructPath(szSysPath, sizeof(szSysPath), "block/");
4133 AssertRCReturn(rc, rc); /* this shall not fail */
4134 if (RTDirExists(szSysPath))
4135 {
4136 rc = vmdkFindSysBlockDevPath(pImage, szSysPath, sizeof(szSysPath), StDrive.st_rdev, pszRawDrive);
4137
4138 /* Now, scan the directories under that again for a partition device
4139 matching the hRawPart device's number: */
4140 if (RT_SUCCESS(rc))
4141 rc = vmdkFindSysBlockDevPath(pImage, szSysPath, sizeof(szSysPath), StPart.st_rdev, pPartDesc->pszRawDevice);
4142
4143 /* Having found the /sys/block/device/partition/ path, we can finally
4144 read the partition attributes and compare with hVol. */
4145 if (RT_SUCCESS(rc))
4146 {
4147 /* partition number: */
4148 int64_t iLnxPartition = 0;
4149 rc = RTLinuxSysFsReadIntFile(10, &iLnxPartition, "%s/partition", szSysPath);
4150 if (RT_SUCCESS(rc) && iLnxPartition != idxPartition)
4151 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4152 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Partition number %RI64, expected %RU32"),
4153 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, iLnxPartition, idxPartition);
4154 /* else: ignore failure? */
4155
4156 /* start offset: */
4157 uint32_t const cbLnxSector = 512; /* It's hardcoded in the Linux kernel */
4158 if (RT_SUCCESS(rc))
4159 {
4160 int64_t offLnxStart = -1;
4161 rc = RTLinuxSysFsReadIntFile(10, &offLnxStart, "%s/start", szSysPath);
4162 offLnxStart *= cbLnxSector;
4163 if (RT_SUCCESS(rc) && offLnxStart != (int64_t)pPartDesc->offStartInVDisk)
4164 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4165 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Start offset %RI64, expected %RU64"),
4166 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, offLnxStart, pPartDesc->offStartInVDisk);
4167 /* else: ignore failure? */
4168 }
4169
4170 /* the size: */
4171 if (RT_SUCCESS(rc))
4172 {
4173 int64_t cbLnxData = -1;
4174 rc = RTLinuxSysFsReadIntFile(10, &cbLnxData, "%s/size", szSysPath);
4175 cbLnxData *= cbLnxSector;
4176 if (RT_SUCCESS(rc) && cbLnxData != (int64_t)pPartDesc->cbData)
4177 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4178 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Size %RI64, expected %RU64"),
4179 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, cbLnxData, pPartDesc->cbData);
4180 /* else: ignore failure? */
4181 }
4182 }
4183 }
4184 /* else: We've got nothing to work on, so only do content comparison. */
4185 }
4186
4187#elif defined(RT_OS_FREEBSD)
4188 char szDriveDevName[256];
4189 char* pszDevName = fdevname_r(RTFileToNative(hRawDrive), szDriveDevName, 256);
4190 if (pszDevName == NULL)
4191 rc = vdIfError(pImage->pIfError, VERR_INVALID_PARAMETER, RT_SRC_POS,
4192 N_("VMDK: Image path: '%s'. '%s' is not a drive path"), pImage->pszFilename, pszRawDrive);
4193 char szPartDevName[256];
4194 if (RT_SUCCESS(rc))
4195 {
4196 pszDevName = fdevname_r(RTFileToNative(hRawPart), szPartDevName, 256);
4197 if (pszDevName == NULL)
4198 rc = vdIfError(pImage->pIfError, VERR_INVALID_PARAMETER, RT_SRC_POS,
4199 N_("VMDK: Image path: '%s'. '%s' is not a partition path"), pImage->pszFilename, pPartDesc->pszRawDevice);
4200 }
4201 if (RT_SUCCESS(rc))
4202 {
4203 gmesh geomMesh;
4204 int err = geom_gettree(&geomMesh);
4205 if (err == 0)
4206 {
4207 /* Find root class containg partitions info */
4208 gclass* pPartClass;
4209 LIST_FOREACH(pPartClass, &geomMesh.lg_class, lg_class)
4210 {
4211 if (RTStrCmp(pPartClass->lg_name, "PART") == 0)
4212 break;
4213 }
4214 if (pPartClass == NULL || RTStrCmp(pPartClass->lg_name, "PART") != 0)
4215 rc = vdIfError(pImage->pIfError, VERR_GENERAL_FAILURE, RT_SRC_POS,
4216 N_("VMDK: Image path: '%s'. 'PART' class not found in the GEOM tree"), pImage->pszFilename);
4217
4218
4219 if (RT_SUCCESS(rc))
4220 {
4221 /* Find provider representing partition device */
4222 uint64_t cbOffset;
4223 uint64_t cbSize;
4224 rc = vmdkFindPartitionParamsByName(pPartClass, szDriveDevName, szPartDevName, &cbOffset, &cbSize);
4225 if (RT_SUCCESS(rc))
4226 {
4227 if (cbOffset != pPartDesc->offStartInVDisk)
4228 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4229 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Start offset %RU64, expected %RU64"),
4230 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, cbOffset, pPartDesc->offStartInVDisk);
4231 if (cbSize != pPartDesc->cbData)
4232 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4233 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Size %RU64, expected %RU64"),
4234 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, cbSize, pPartDesc->cbData);
4235 }
4236 else
4237 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4238 N_("VMDK: Image path: '%s'. Error getting geom provider for the partition '%s' of the drive '%s' in the GEOM tree: %Rrc"),
4239 pImage->pszFilename, pPartDesc->pszRawDevice, pszRawDrive, rc);
4240 }
4241
4242 geom_deletetree(&geomMesh);
4243 }
4244 else
4245 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(err), RT_SRC_POS,
4246 N_("VMDK: Image path: '%s'. geom_gettree failed: %d"), pImage->pszFilename, err);
4247 }
4248
4249#elif defined(RT_OS_SOLARIS)
4250 RT_NOREF(hVol);
4251
4252 dk_cinfo dkiDriveInfo;
4253 dk_cinfo dkiPartInfo;
4254 if (ioctl(RTFileToNative(hRawDrive), DKIOCINFO, (caddr_t)&dkiDriveInfo) == -1)
4255 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4256 N_("VMDK: Image path: '%s'. DKIOCINFO failed on '%s': %d"), pImage->pszFilename, pszRawDrive, errno);
4257 else if (ioctl(RTFileToNative(hRawPart), DKIOCINFO, (caddr_t)&dkiPartInfo) == -1)
4258 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4259 N_("VMDK: Image path: '%s'. DKIOCINFO failed on '%s': %d"), pImage->pszFilename, pszRawDrive, errno);
4260 else if ( dkiDriveInfo.dki_ctype != dkiPartInfo.dki_ctype
4261 || dkiDriveInfo.dki_cnum != dkiPartInfo.dki_cnum
4262 || dkiDriveInfo.dki_addr != dkiPartInfo.dki_addr
4263 || dkiDriveInfo.dki_unit != dkiPartInfo.dki_unit
4264 || dkiDriveInfo.dki_slave != dkiPartInfo.dki_slave)
4265 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4266 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s' (%#x != %#x || %#x != %#x || %#x != %#x || %#x != %#x || %#x != %#x)"),
4267 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive,
4268 dkiDriveInfo.dki_ctype, dkiPartInfo.dki_ctype, dkiDriveInfo.dki_cnum, dkiPartInfo.dki_cnum,
4269 dkiDriveInfo.dki_addr, dkiPartInfo.dki_addr, dkiDriveInfo.dki_unit, dkiPartInfo.dki_unit,
4270 dkiDriveInfo.dki_slave, dkiPartInfo.dki_slave);
4271 else
4272 {
4273 uint64_t cbOffset = 0;
4274 uint64_t cbSize = 0;
4275 dk_gpt *pEfi = NULL;
4276 int idxEfiPart = efi_alloc_and_read(RTFileToNative(hRawPart), &pEfi);
4277 if (idxEfiPart >= 0)
4278 {
4279 if ((uint32_t)dkiPartInfo.dki_partition + 1 == idxPartition)
4280 {
4281 cbOffset = pEfi->efi_parts[idxEfiPart].p_start * pEfi->efi_lbasize;
4282 cbSize = pEfi->efi_parts[idxEfiPart].p_size * pEfi->efi_lbasize;
4283 }
4284 else
4285 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4286 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s' (%#x != %#x)"),
4287 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive,
4288 idxPartition, (uint32_t)dkiPartInfo.dki_partition + 1);
4289 efi_free(pEfi);
4290 }
4291 else
4292 {
4293 /*
4294 * Manual says the efi_alloc_and_read returns VT_EINVAL if no EFI partition table found.
4295 * Actually, the function returns any error, e.g. VT_ERROR. Thus, we are not sure, is it
4296 * real error or just no EFI table found. Therefore, let's try to obtain partition info
4297 * using another way. If there is an error, it returns errno which will be handled below.
4298 */
4299
4300 uint32_t numPartition = (uint32_t)dkiPartInfo.dki_partition;
4301 if (numPartition > NDKMAP)
4302 numPartition -= NDKMAP;
4303 if (numPartition != idxPartition)
4304 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4305 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s' (%#x != %#x)"),
4306 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive,
4307 idxPartition, numPartition);
4308 else
4309 {
4310 dk_minfo_ext mediaInfo;
4311 if (ioctl(RTFileToNative(hRawPart), DKIOCGMEDIAINFOEXT, (caddr_t)&mediaInfo) == -1)
4312 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4313 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s'. Can not obtain partition info: %d"),
4314 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, errno);
4315 else
4316 {
4317 extpart_info extPartInfo;
4318 if (ioctl(RTFileToNative(hRawPart), DKIOCEXTPARTINFO, (caddr_t)&extPartInfo) != -1)
4319 {
4320 cbOffset = (uint64_t)extPartInfo.p_start * mediaInfo.dki_lbsize;
4321 cbSize = (uint64_t)extPartInfo.p_length * mediaInfo.dki_lbsize;
4322 }
4323 else
4324 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4325 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s'. Can not obtain partition info: %d"),
4326 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, errno);
4327 }
4328 }
4329 }
4330 if (RT_SUCCESS(rc) && cbOffset != pPartDesc->offStartInVDisk)
4331 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4332 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Start offset %RI64, expected %RU64"),
4333 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, cbOffset, pPartDesc->offStartInVDisk);
4334
4335 if (RT_SUCCESS(rc) && cbSize != pPartDesc->cbData)
4336 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4337 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Size %RI64, expected %RU64"),
4338 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, cbSize, pPartDesc->cbData);
4339 }
4340
4341#elif defined(RT_OS_DARWIN)
4342 /* Stat the drive get its device number. */
4343 struct stat StDrive;
4344 if (fstat((int)RTFileToNative(hRawDrive), &StDrive) != 0)
4345 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4346 N_("VMDK: Image path: '%s'. fstat failed on '%s' (errno=%d)"), pImage->pszFilename, pszRawDrive, errno);
4347 else
4348 {
4349 if (ioctl(RTFileToNative(hRawPart), DKIOCLOCKPHYSICALEXTENTS, NULL) == -1)
4350 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4351 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s': Unable to lock the partition (errno=%d)"),
4352 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, errno);
4353 else
4354 {
4355 uint32_t cbBlockSize = 0;
4356 uint64_t cbOffset = 0;
4357 uint64_t cbSize = 0;
4358 if (ioctl(RTFileToNative(hRawPart), DKIOCGETBLOCKSIZE, (caddr_t)&cbBlockSize) == -1)
4359 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4360 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s': Unable to obtain the sector size of the partition (errno=%d)"),
4361 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, errno);
4362 else if (ioctl(RTFileToNative(hRawPart), DKIOCGETBASE, (caddr_t)&cbOffset) == -1)
4363 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4364 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s': Unable to obtain the start offset of the partition (errno=%d)"),
4365 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, errno);
4366 else if (ioctl(RTFileToNative(hRawPart), DKIOCGETBLOCKCOUNT, (caddr_t)&cbSize) == -1)
4367 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4368 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s': Unable to obtain the size of the partition (errno=%d)"),
4369 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, errno);
4370 else
4371 {
4372 cbSize *= (uint64_t)cbBlockSize;
4373 dk_physical_extent_t dkPartExtent = {0};
4374 dkPartExtent.offset = 0;
4375 dkPartExtent.length = cbSize;
4376 if (ioctl(RTFileToNative(hRawPart), DKIOCGETPHYSICALEXTENT, (caddr_t)&dkPartExtent) == -1)
4377 rc = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4378 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s': Unable to obtain partition info (errno=%d)"),
4379 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, errno);
4380 else
4381 {
4382 if (dkPartExtent.dev != StDrive.st_rdev)
4383 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4384 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Drive does not contain the partition"),
4385 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive);
4386 else if (cbOffset != pPartDesc->offStartInVDisk)
4387 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4388 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Start offset %RU64, expected %RU64"),
4389 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, cbOffset, pPartDesc->offStartInVDisk);
4390 else if (cbSize != pPartDesc->cbData)
4391 rc = vdIfError(pImage->pIfError, VERR_MISMATCH, RT_SRC_POS,
4392 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s': Size %RU64, expected %RU64"),
4393 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, cbSize, pPartDesc->cbData);
4394 }
4395 }
4396
4397 if (ioctl(RTFileToNative(hRawPart), DKIOCUNLOCKPHYSICALEXTENTS, NULL) == -1)
4398 {
4399 int rc2 = vdIfError(pImage->pIfError, RTErrConvertFromErrno(errno), RT_SRC_POS,
4400 N_("VMDK: Image path: '%s'. Partition #%u number ('%s') verification failed on '%s': Unable to unlock the partition (errno=%d)"),
4401 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, errno);
4402 if (RT_SUCCESS(rc))
4403 rc = rc2;
4404 }
4405 }
4406 }
4407
4408#else
4409 RT_NOREF(hVol); /* PORTME */
4410 rc = VERR_NOT_SUPPORTED;
4411#endif
4412 if (RT_SUCCESS(rc))
4413 {
4414 /*
4415 * Compare the first 32 sectors of the partition.
4416 *
4417 * This might not be conclusive, but for partitions formatted with the more
4418 * common file systems it should be as they have a superblock copy at or near
4419 * the start of the partition (fat, fat32, ntfs, and ext4 does at least).
4420 */
4421 size_t const cbToCompare = (size_t)RT_MIN(pPartDesc->cbData / cbSector, 32) * cbSector;
4422 uint8_t *pbSector1 = (uint8_t *)RTMemTmpAlloc(cbToCompare * 2);
4423 if (pbSector1 != NULL)
4424 {
4425 uint8_t *pbSector2 = pbSector1 + cbToCompare;
4426
4427 /* Do the comparing, we repeat if it fails and the data might be volatile. */
4428 uint64_t uPrevCrc1 = 0;
4429 uint64_t uPrevCrc2 = 0;
4430 uint32_t cStable = 0;
4431 for (unsigned iTry = 0; iTry < 256; iTry++)
4432 {
4433 rc = RTFileReadAt(hRawDrive, pPartDesc->offStartInVDisk, pbSector1, cbToCompare, NULL);
4434 if (RT_SUCCESS(rc))
4435 {
4436 rc = RTFileReadAt(hRawPart, pPartDesc->offStartInDevice, pbSector2, cbToCompare, NULL);
4437 if (RT_SUCCESS(rc))
4438 {
4439 if (memcmp(pbSector1, pbSector2, cbToCompare) != 0)
4440 {
4441 rc = VERR_MISMATCH;
4442
4443 /* Do data stability checks before repeating: */
4444 uint64_t const uCrc1 = RTCrc64(pbSector1, cbToCompare);
4445 uint64_t const uCrc2 = RTCrc64(pbSector2, cbToCompare);
4446 if ( uPrevCrc1 != uCrc1
4447 || uPrevCrc2 != uCrc2)
4448 cStable = 0;
4449 else if (++cStable > 4)
4450 break;
4451 uPrevCrc1 = uCrc1;
4452 uPrevCrc2 = uCrc2;
4453 continue;
4454 }
4455 rc = VINF_SUCCESS;
4456 }
4457 else
4458 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4459 N_("VMDK: Image path: '%s'. Error reading %zu bytes from '%s' at offset %RU64 (%Rrc)"),
4460 pImage->pszFilename, cbToCompare, pPartDesc->pszRawDevice, pPartDesc->offStartInDevice, rc);
4461 }
4462 else
4463 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4464 N_("VMDK: Image path: '%s'. Error reading %zu bytes from '%s' at offset %RU64 (%Rrc)"),
4465 pImage->pszFilename, cbToCompare, pszRawDrive, pPartDesc->offStartInVDisk, rc);
4466 break;
4467 }
4468 if (rc == VERR_MISMATCH)
4469 {
4470 /* Find the first mismatching bytes: */
4471 size_t offMissmatch = 0;
4472 while (offMissmatch < cbToCompare && pbSector1[offMissmatch] == pbSector2[offMissmatch])
4473 offMissmatch++;
4474 int cbSample = (int)RT_MIN(cbToCompare - offMissmatch, 16);
4475
4476 if (cStable > 0)
4477 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4478 N_("VMDK: Image path: '%s'. Partition #%u path ('%s') verification failed on '%s' (cStable=%d @%#zx: %.*Rhxs vs %.*Rhxs)"),
4479 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive, cStable,
4480 offMissmatch, cbSample, &pbSector1[offMissmatch], cbSample, &pbSector2[offMissmatch]);
4481 else
4482 {
4483 LogRel(("VMDK: Image path: '%s'. Partition #%u path ('%s') verification undecided on '%s' because of unstable data! (@%#zx: %.*Rhxs vs %.*Rhxs)\n",
4484 pImage->pszFilename, idxPartition, pPartDesc->pszRawDevice, pszRawDrive,
4485 offMissmatch, cbSample, &pbSector1[offMissmatch], cbSample, &pbSector2[offMissmatch]));
4486 rc = -rc;
4487 }
4488 }
4489
4490 RTMemTmpFree(pbSector1);
4491 }
4492 else
4493 rc = vdIfError(pImage->pIfError, VERR_NO_TMP_MEMORY, RT_SRC_POS,
4494 N_("VMDK: Image path: '%s'. Failed to allocate %zu bytes for a temporary read buffer\n"),
4495 pImage->pszFilename, cbToCompare * 2);
4496 }
4497 RTFileClose(hRawPart);
4498 return rc;
4499}
4500
4501#ifdef RT_OS_WINDOWS
4502/**
4503 * Construct the device name for the given partition number.
4504 */
4505static int vmdkRawDescWinMakePartitionName(PVMDKIMAGE pImage, const char *pszRawDrive, RTFILE hRawDrive, uint32_t idxPartition,
4506 char **ppszRawPartition)
4507{
4508 int rc = VINF_SUCCESS;
4509 DWORD cbReturned = 0;
4510 STORAGE_DEVICE_NUMBER DevNum;
4511 RT_ZERO(DevNum);
4512 if (DeviceIoControl((HANDLE)RTFileToNative(hRawDrive), IOCTL_STORAGE_GET_DEVICE_NUMBER,
4513 NULL /*pvInBuffer*/, 0 /*cbInBuffer*/, &DevNum, sizeof(DevNum), &cbReturned, NULL /*pOverlapped*/))
4514 RTStrAPrintf(ppszRawPartition, "\\\\.\\Harddisk%uPartition%u", DevNum.DeviceNumber, idxPartition);
4515 else
4516 rc = vdIfError(pImage->pIfError, RTErrConvertFromWin32(GetLastError()), RT_SRC_POS,
4517 N_("VMDK: Image path: '%s'. IOCTL_STORAGE_GET_DEVICE_NUMBER failed on '%s': %u"),
4518 pImage->pszFilename, pszRawDrive, GetLastError());
4519 return rc;
4520}
4521#endif /* RT_OS_WINDOWS */
4522
4523/**
4524 * Worker for vmdkMakeRawDescriptor that adds partition descriptors when the
4525 * 'Partitions' configuration value is present.
4526 *
4527 * @returns VBox status code, error message has been set on failure.
4528 *
4529 * @note Caller is assumed to clean up @a pRawDesc and release
4530 * @a *phVolToRelease.
4531 * @internal
4532 */
4533static int vmdkRawDescDoPartitions(PVMDKIMAGE pImage, RTDVM hVolMgr, PVDISKRAW pRawDesc,
4534 RTFILE hRawDrive, const char *pszRawDrive, uint32_t cbSector,
4535 uint32_t fPartitions, uint32_t fPartitionsReadOnly, bool fRelative,
4536 PRTDVMVOLUME phVolToRelease)
4537{
4538 *phVolToRelease = NIL_RTDVMVOLUME;
4539
4540 /* Check sanity/understanding. */
4541 Assert(fPartitions);
4542 Assert((fPartitions & fPartitionsReadOnly) == fPartitionsReadOnly); /* RO should be a sub-set */
4543
4544 /*
4545 * Allocate on descriptor for each volume up front.
4546 */
4547 uint32_t const cVolumes = RTDvmMapGetValidVolumes(hVolMgr);
4548
4549 PVDISKRAWPARTDESC paPartDescs = NULL;
4550 int rc = vmdkRawDescAppendPartDesc(pImage, pRawDesc, cVolumes, &paPartDescs);
4551 AssertRCReturn(rc, rc);
4552
4553 /*
4554 * Enumerate the partitions (volumes) on the disk and create descriptors for each of them.
4555 */
4556 uint32_t fPartitionsLeft = fPartitions;
4557 RTDVMVOLUME hVol = NIL_RTDVMVOLUME; /* the current volume, needed for getting the next. */
4558 for (uint32_t i = 0; i < cVolumes; i++)
4559 {
4560 /*
4561 * Get the next/first volume and release the current.
4562 */
4563 RTDVMVOLUME hVolNext = NIL_RTDVMVOLUME;
4564 if (i == 0)
4565 rc = RTDvmMapQueryFirstVolume(hVolMgr, &hVolNext);
4566 else
4567 rc = RTDvmMapQueryNextVolume(hVolMgr, hVol, &hVolNext);
4568 if (RT_FAILURE(rc))
4569 return vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4570 N_("VMDK: Image path: '%s'. Volume enumeration failed at volume #%u on '%s' (%Rrc)"),
4571 pImage->pszFilename, i, pszRawDrive, rc);
4572 uint32_t cRefs = RTDvmVolumeRelease(hVol);
4573 Assert(cRefs != UINT32_MAX); RT_NOREF(cRefs);
4574 *phVolToRelease = hVol = hVolNext;
4575
4576 /*
4577 * Depending on the fPartitions selector and associated read-only mask,
4578 * the guest either gets read-write or read-only access (bits set)
4579 * or no access (selector bit clear, access directed to the VMDK).
4580 */
4581 paPartDescs[i].cbData = RTDvmVolumeGetSize(hVol);
4582
4583 uint64_t offVolumeEndIgnored = 0;
4584 rc = RTDvmVolumeQueryRange(hVol, &paPartDescs[i].offStartInVDisk, &offVolumeEndIgnored);
4585 if (RT_FAILURE(rc))
4586 return vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4587 N_("VMDK: Image path: '%s'. Failed to get location of volume #%u on '%s' (%Rrc)"),
4588 pImage->pszFilename, i, pszRawDrive, rc);
4589 Assert(paPartDescs[i].cbData == offVolumeEndIgnored + 1 - paPartDescs[i].offStartInVDisk);
4590
4591 /* Note! The index must match IHostDrivePartition::number. */
4592 uint32_t idxPartition = RTDvmVolumeGetIndex(hVol, RTDVMVOLIDX_HOST);
4593 if ( idxPartition < 32
4594 && (fPartitions & RT_BIT_32(idxPartition)))
4595 {
4596 fPartitionsLeft &= ~RT_BIT_32(idxPartition);
4597 if (fPartitionsReadOnly & RT_BIT_32(idxPartition))
4598 paPartDescs[i].uFlags |= VDISKRAW_READONLY;
4599
4600 if (!fRelative)
4601 {
4602 /*
4603 * Accessing the drive thru the main device node (pRawDesc->pszRawDisk).
4604 */
4605 paPartDescs[i].offStartInDevice = paPartDescs[i].offStartInVDisk;
4606 paPartDescs[i].pszRawDevice = RTStrDup(pszRawDrive);
4607 AssertPtrReturn(paPartDescs[i].pszRawDevice, VERR_NO_STR_MEMORY);
4608 }
4609 else
4610 {
4611 /*
4612 * Relative means access the partition data via the device node for that
4613 * partition, allowing the sysadmin/OS to allow a user access to individual
4614 * partitions without necessarily being able to compromise the host OS.
4615 * Obviously, the creation of the VMDK requires read access to the main
4616 * device node for the drive, but that's a one-time thing and can be done
4617 * by the sysadmin. Here data starts at offset zero in the device node.
4618 */
4619 paPartDescs[i].offStartInDevice = 0;
4620
4621#if defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
4622 /* /dev/rdisk1 -> /dev/rdisk1s2 (s=slice) */
4623 RTStrAPrintf(&paPartDescs[i].pszRawDevice, "%ss%u", pszRawDrive, idxPartition);
4624#elif defined(RT_OS_LINUX)
4625 /* Two naming schemes here: /dev/nvme0n1 -> /dev/nvme0n1p1; /dev/sda -> /dev/sda1 */
4626 RTStrAPrintf(&paPartDescs[i].pszRawDevice,
4627 RT_C_IS_DIGIT(pszRawDrive[strlen(pszRawDrive) - 1]) ? "%sp%u" : "%s%u", pszRawDrive, idxPartition);
4628#elif defined(RT_OS_WINDOWS)
4629 rc = vmdkRawDescWinMakePartitionName(pImage, pszRawDrive, hRawDrive, idxPartition, &paPartDescs[i].pszRawDevice);
4630 AssertRCReturn(rc, rc);
4631#elif defined(RT_OS_SOLARIS)
4632 if (pRawDesc->enmPartitioningType == VDISKPARTTYPE_MBR)
4633 {
4634 /*
4635 * MBR partitions have device nodes in form /dev/(r)dsk/cXtYdZpK
4636 * where X is the controller,
4637 * Y is target (SCSI device number),
4638 * Z is disk number,
4639 * K is partition number,
4640 * where p0 is the whole disk
4641 * p1-pN are the partitions of the disk
4642 */
4643 const char *pszRawDrivePath = pszRawDrive;
4644 char szDrivePath[RTPATH_MAX];
4645 size_t cbRawDrive = strlen(pszRawDrive);
4646 if ( cbRawDrive > 1 && strcmp(&pszRawDrive[cbRawDrive - 2], "p0") == 0)
4647 {
4648 memcpy(szDrivePath, pszRawDrive, cbRawDrive - 2);
4649 szDrivePath[cbRawDrive - 2] = '\0';
4650 pszRawDrivePath = szDrivePath;
4651 }
4652 RTStrAPrintf(&paPartDescs[i].pszRawDevice, "%sp%u", pszRawDrivePath, idxPartition);
4653 }
4654 else /* GPT */
4655 {
4656 /*
4657 * GPT partitions have device nodes in form /dev/(r)dsk/cXtYdZsK
4658 * where X is the controller,
4659 * Y is target (SCSI device number),
4660 * Z is disk number,
4661 * K is partition number, zero based. Can be only from 0 to 6.
4662 * Thus, only partitions numbered 0 through 6 have device nodes.
4663 */
4664 if (idxPartition > 7)
4665 return vdIfError(pImage->pIfError, VERR_INVALID_PARAMETER, RT_SRC_POS,
4666 N_("VMDK: Image path: '%s'. the partition #%u on '%s' has no device node and can not be specified with 'Relative' property"),
4667 pImage->pszFilename, idxPartition, pszRawDrive);
4668 RTStrAPrintf(&paPartDescs[i].pszRawDevice, "%ss%u", pszRawDrive, idxPartition - 1);
4669 }
4670#else
4671 AssertFailedReturn(VERR_INTERNAL_ERROR_4); /* The option parsing code should have prevented this - PORTME */
4672#endif
4673 AssertPtrReturn(paPartDescs[i].pszRawDevice, VERR_NO_STR_MEMORY);
4674
4675 rc = vmdkRawDescVerifyPartitionPath(pImage, &paPartDescs[i], idxPartition, pszRawDrive, hRawDrive, cbSector, hVol);
4676 AssertRCReturn(rc, rc);
4677 }
4678 }
4679 else
4680 {
4681 /* Not accessible to the guest. */
4682 paPartDescs[i].offStartInDevice = 0;
4683 paPartDescs[i].pszRawDevice = NULL;
4684 }
4685 } /* for each volume */
4686
4687 RTDvmVolumeRelease(hVol);
4688 *phVolToRelease = NIL_RTDVMVOLUME;
4689
4690 /*
4691 * Check that we found all the partitions the user selected.
4692 */
4693 if (fPartitionsLeft)
4694 {
4695 char szLeft[3 * sizeof(fPartitions) * 8];
4696 size_t cchLeft = 0;
4697 for (unsigned i = 0; i < sizeof(fPartitions) * 8; i++)
4698 if (fPartitionsLeft & RT_BIT_32(i))
4699 cchLeft += RTStrPrintf(&szLeft[cchLeft], sizeof(szLeft) - cchLeft, cchLeft ? "%u" : ",%u", i);
4700 return vdIfError(pImage->pIfError, VERR_INVALID_PARAMETER, RT_SRC_POS,
4701 N_("VMDK: Image path: '%s'. Not all the specified partitions for drive '%s' was found: %s"),
4702 pImage->pszFilename, pszRawDrive, szLeft);
4703 }
4704
4705 return VINF_SUCCESS;
4706}
4707
4708/**
4709 * Worker for vmdkMakeRawDescriptor that adds partition descriptors with copies
4710 * of the partition tables and associated padding areas when the 'Partitions'
4711 * configuration value is present.
4712 *
4713 * The guest is not allowed access to the partition tables, however it needs
4714 * them to be able to access the drive. So, create descriptors for each of the
4715 * tables and attach the current disk content. vmdkCreateRawImage() will later
4716 * write the content to the VMDK. Any changes the guest later makes to the
4717 * partition tables will then go to the VMDK copy, rather than the host drive.
4718 *
4719 * @returns VBox status code, error message has been set on failure.
4720 *
4721 * @note Caller is assumed to clean up @a pRawDesc
4722 * @internal
4723 */
4724static int vmdkRawDescDoCopyPartitionTables(PVMDKIMAGE pImage, RTDVM hVolMgr, PVDISKRAW pRawDesc,
4725 const char *pszRawDrive, RTFILE hRawDrive, void *pvBootSector, size_t cbBootSector)
4726{
4727 /*
4728 * Query the locations.
4729 */
4730 /* Determin how many locations there are: */
4731 size_t cLocations = 0;
4732 int rc = RTDvmMapQueryTableLocations(hVolMgr, RTDVMMAPQTABLOC_F_INCLUDE_LEGACY, NULL, 0, &cLocations);
4733 if (rc != VERR_BUFFER_OVERFLOW)
4734 return vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4735 N_("VMDK: Image path: '%s'. RTDvmMapQueryTableLocations failed on '%s' (%Rrc)"),
4736 pImage->pszFilename, pszRawDrive, rc);
4737 AssertReturn(cLocations > 0 && cLocations < _16M, VERR_INTERNAL_ERROR_5);
4738
4739 /* We can allocate the partition descriptors here to save an intentation level. */
4740 PVDISKRAWPARTDESC paPartDescs = NULL;
4741 rc = vmdkRawDescAppendPartDesc(pImage, pRawDesc, (uint32_t)cLocations, &paPartDescs);
4742 AssertRCReturn(rc, rc);
4743
4744 /* Allocate the result table and repeat the location table query: */
4745 PRTDVMTABLELOCATION paLocations = (PRTDVMTABLELOCATION)RTMemAllocZ(sizeof(paLocations[0]) * cLocations);
4746 if (!paLocations)
4747 return vdIfError(pImage->pIfError, VERR_NO_MEMORY, RT_SRC_POS, N_("VMDK: Image path: '%s'. Failed to allocate %zu bytes"),
4748 pImage->pszFilename, sizeof(paLocations[0]) * cLocations);
4749 rc = RTDvmMapQueryTableLocations(hVolMgr, RTDVMMAPQTABLOC_F_INCLUDE_LEGACY, paLocations, cLocations, NULL);
4750 if (RT_SUCCESS(rc))
4751 {
4752 /*
4753 * Translate them into descriptors.
4754 *
4755 * We restrict the amount of partition alignment padding to 4MiB as more
4756 * will just be a waste of space. The use case for including the padding
4757 * are older boot loaders and boot manager (including one by a team member)
4758 * that put data and code in the 62 sectors between the MBR and the first
4759 * partition (total of 63). Later CHS was abandond and partition started
4760 * being aligned on power of two sector boundraries (typically 64KiB or
4761 * 1MiB depending on the media size).
4762 */
4763 for (size_t i = 0; i < cLocations && RT_SUCCESS(rc); i++)
4764 {
4765 Assert(paLocations[i].cb > 0);
4766 if (paLocations[i].cb <= _64M)
4767 {
4768 /* Create the partition descriptor entry: */
4769 //paPartDescs[i].pszRawDevice = NULL;
4770 //paPartDescs[i].offStartInDevice = 0;
4771 //paPartDescs[i].uFlags = 0;
4772 paPartDescs[i].offStartInVDisk = paLocations[i].off;
4773 paPartDescs[i].cbData = paLocations[i].cb;
4774 if (paPartDescs[i].cbData < _4M)
4775 paPartDescs[i].cbData = RT_MIN(paPartDescs[i].cbData + paLocations[i].cbPadding, _4M);
4776 paPartDescs[i].pvPartitionData = RTMemAllocZ((size_t)paPartDescs[i].cbData);
4777 if (paPartDescs[i].pvPartitionData)
4778 {
4779 /* Read the content from the drive: */
4780 rc = RTFileReadAt(hRawDrive, paPartDescs[i].offStartInVDisk, paPartDescs[i].pvPartitionData,
4781 (size_t)paPartDescs[i].cbData, NULL);
4782 if (RT_SUCCESS(rc))
4783 {
4784 /* Do we have custom boot sector code? */
4785 if (pvBootSector && cbBootSector && paPartDescs[i].offStartInVDisk == 0)
4786 {
4787 /* Note! Old code used to quietly drop the bootsector if it was considered too big.
4788 Instead we fail as we weren't able to do what the user requested us to do.
4789 Better if the user knows than starts questioning why the guest isn't
4790 booting as expected. */
4791 if (cbBootSector <= paPartDescs[i].cbData)
4792 memcpy(paPartDescs[i].pvPartitionData, pvBootSector, cbBootSector);
4793 else
4794 rc = vdIfError(pImage->pIfError, VERR_TOO_MUCH_DATA, RT_SRC_POS,
4795 N_("VMDK: Image path: '%s'. The custom boot sector is too big: %zu bytes, %RU64 bytes available"),
4796 pImage->pszFilename, cbBootSector, paPartDescs[i].cbData);
4797 }
4798 }
4799 else
4800 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4801 N_("VMDK: Image path: '%s'. Failed to read partition at off %RU64 length %zu from '%s' (%Rrc)"),
4802 pImage->pszFilename, paPartDescs[i].offStartInVDisk,
4803 (size_t)paPartDescs[i].cbData, pszRawDrive, rc);
4804 }
4805 else
4806 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4807 N_("VMDK: Image path: '%s'. Failed to allocate %zu bytes for copying the partition table at off %RU64"),
4808 pImage->pszFilename, (size_t)paPartDescs[i].cbData, paPartDescs[i].offStartInVDisk);
4809 }
4810 else
4811 rc = vdIfError(pImage->pIfError, VERR_TOO_MUCH_DATA, RT_SRC_POS,
4812 N_("VMDK: Image path: '%s'. Partition table #%u at offset %RU64 in '%s' is to big: %RU64 bytes"),
4813 pImage->pszFilename, i, paLocations[i].off, pszRawDrive, paLocations[i].cb);
4814 }
4815 }
4816 else
4817 rc = vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4818 N_("VMDK: Image path: '%s'. RTDvmMapQueryTableLocations failed on '%s' (%Rrc)"),
4819 pImage->pszFilename, pszRawDrive, rc);
4820 RTMemFree(paLocations);
4821 return rc;
4822}
4823
4824/**
4825 * Opens the volume manager for the raw drive when in selected-partition mode.
4826 *
4827 * @param pImage The VMDK image (for errors).
4828 * @param hRawDrive The raw drive handle.
4829 * @param pszRawDrive The raw drive device path (for errors).
4830 * @param cbSector The sector size.
4831 * @param phVolMgr Where to return the handle to the volume manager on
4832 * success.
4833 * @returns VBox status code, errors have been reported.
4834 * @internal
4835 */
4836static int vmdkRawDescOpenVolMgr(PVMDKIMAGE pImage, RTFILE hRawDrive, const char *pszRawDrive, uint32_t cbSector, PRTDVM phVolMgr)
4837{
4838 *phVolMgr = NIL_RTDVM;
4839
4840 RTVFSFILE hVfsFile = NIL_RTVFSFILE;
4841 int rc = RTVfsFileFromRTFile(hRawDrive, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE, true /*fLeaveOpen*/, &hVfsFile);
4842 if (RT_FAILURE(rc))
4843 return vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4844 N_("VMDK: Image path: '%s'. RTVfsFileFromRTFile failed for '%s' handle (%Rrc)"),
4845 pImage->pszFilename, pszRawDrive, rc);
4846
4847 RTDVM hVolMgr = NIL_RTDVM;
4848 rc = RTDvmCreate(&hVolMgr, hVfsFile, cbSector, 0 /*fFlags*/);
4849
4850 RTVfsFileRelease(hVfsFile);
4851
4852 if (RT_FAILURE(rc))
4853 return vdIfError(pImage->pIfError, rc, RT_SRC_POS,
4854 N_("VMDK: Image path: '%s'. Failed to create volume manager instance for '%s' (%Rrc)"),
4855 pImage->pszFilename, pszRawDrive, rc);
4856
4857 rc = RTDvmMapOpen(hVolMgr);
4858 if (RT_SUCCESS(rc))
4859 {
4860 *phVolMgr = hVolMgr;
4861 return VINF_SUCCESS;
4862 }
4863 RTDvmRelease(hVolMgr);
4864 return vdIfError(pImage->pIfError, rc, RT_SRC_POS, N_("VMDK: Image path: '%s'. RTDvmMapOpen failed for '%s' (%Rrc)"),
4865 pImage->pszFilename, pszRawDrive, rc);
4866}
4867
4868/**
4869 * Opens the raw drive device and get the sizes for it.
4870 *
4871 * @param pImage The image (for error reporting).
4872 * @param pszRawDrive The device/whatever to open.
4873 * @param phRawDrive Where to return the file handle.
4874 * @param pcbRawDrive Where to return the size.
4875 * @param pcbSector Where to return the sector size.
4876 * @returns IPRT status code, errors have been reported.
4877 * @internal
4878 */
4879static int vmkdRawDescOpenDevice(PVMDKIMAGE pImage, const char *pszRawDrive,
4880 PRTFILE phRawDrive, uint64_t *pcbRawDrive, uint32_t *pcbSector)
4881{
4882 /*
4883 * Open the device for the raw drive.
4884 */
4885 RTFILE hRawDrive = NIL_RTFILE;
4886 int rc = RTFileOpen(&hRawDrive, pszRawDrive, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
4887 if (RT_FAILURE(