VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/testcase/tstVD.cpp@ 13340

Last change on this file since 13340 was 13340, checked in by vboxsync, 16 years ago

#3230: Memory leaks fixed in VHD backend

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 33.2 KB
Line 
1/** @file
2 *
3 * Simple VBox HDD container test utility.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22#include <VBox/err.h>
23#include <VBox/VBoxHDD-new.h>
24#include <iprt/dir.h>
25#include <iprt/string.h>
26#include <iprt/stream.h>
27#include <iprt/file.h>
28#include <iprt/mem.h>
29#include <iprt/initterm.h>
30#include <iprt/rand.h>
31#include "stdio.h"
32#include "stdlib.h"
33
34#define VHD_TEST
35#define VDI_TEST
36#define VMDK_TEST
37
38/*******************************************************************************
39* Global Variables *
40*******************************************************************************/
41/** The error count. */
42unsigned g_cErrors = 0;
43
44
45static void tstVDError(void *pvUser, int rc, RT_SRC_POS_DECL,
46 const char *pszFormat, va_list va)
47{
48 g_cErrors++;
49 RTPrintf("tstVD: Error %Vrc at %s:%u (%s): ", rc, RT_SRC_POS_ARGS);
50 RTPrintfV(pszFormat, va);
51 RTPrintf("\n");
52}
53
54
55static int tstVDCreateDelete(const char *pszBackend, const char *pszFilename,
56 uint64_t cbSize, VDIMAGETYPE enmType,
57 unsigned uFlags, bool fDelete)
58{
59 int rc;
60 PVBOXHDD pVD = NULL;
61 PDMMEDIAGEOMETRY PCHS = { 0, 0, 0 };
62 PDMMEDIAGEOMETRY LCHS = { 0, 0, 0 };
63 PVDINTERFACE pVDIfs = NULL;
64 VDINTERFACE VDIError;
65 VDINTERFACEERROR VDIErrorCallbacks;
66
67#define CHECK(str) \
68 do \
69 { \
70 RTPrintf("%s rc=%Vrc\n", str, rc); \
71 if (VBOX_FAILURE(rc)) \
72 { \
73 VDDestroy(pVD); \
74 return rc; \
75 } \
76 } while (0)
77
78 /* Create error interface. */
79 VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
80 VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
81 VDIErrorCallbacks.pfnError = tstVDError;
82
83 rc = VDInterfaceAdd(&VDIError, "tstVD_Error", VDINTERFACETYPE_ERROR, &VDIErrorCallbacks,
84 NULL, &pVDIfs);
85 AssertRC(rc);
86
87 rc = VDCreate(&VDIError, &pVD);
88 CHECK("VDCreate()");
89
90 rc = VDCreateBase(pVD, pszBackend, pszFilename, enmType, cbSize,
91 uFlags, "Test image", &PCHS, &LCHS, NULL,
92 VD_OPEN_FLAGS_NORMAL, NULL, NULL);
93 CHECK("VDCreateBase()");
94
95 VDDumpImages(pVD);
96
97 VDClose(pVD, fDelete);
98 if (fDelete)
99 {
100 RTFILE File;
101 rc = RTFileOpen(&File, pszFilename, RTFILE_O_READ);
102 if (VBOX_SUCCESS(rc))
103 {
104 RTFileClose(File);
105 return VERR_INTERNAL_ERROR;
106 }
107 }
108
109 VDDestroy(pVD);
110#undef CHECK
111 return 0;
112}
113
114static int tstVDOpenDelete(const char *pszBackend, const char *pszFilename)
115{
116 int rc;
117 PVBOXHDD pVD = NULL;
118 PDMMEDIAGEOMETRY PCHS = { 0, 0, 0 };
119 PDMMEDIAGEOMETRY LCHS = { 0, 0, 0 };
120 PVDINTERFACE pVDIfs = NULL;
121 VDINTERFACE VDIError;
122 VDINTERFACEERROR VDIErrorCallbacks;
123
124#define CHECK(str) \
125 do \
126 { \
127 RTPrintf("%s rc=%Vrc\n", str, rc); \
128 if (VBOX_FAILURE(rc)) \
129 { \
130 VDDestroy(pVD); \
131 return rc; \
132 } \
133 } while (0)
134
135 /* Create error interface. */
136 VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
137 VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
138 VDIErrorCallbacks.pfnError = tstVDError;
139
140 rc = VDInterfaceAdd(&VDIError, "tstVD_Error", VDINTERFACETYPE_ERROR, &VDIErrorCallbacks,
141 NULL, &pVDIfs);
142 AssertRC(rc);
143
144 rc = VDCreate(&VDIError, &pVD);
145 CHECK("VDCreate()");
146
147 rc = VDOpen(pVD, pszBackend, pszFilename, VD_OPEN_FLAGS_NORMAL, NULL);
148 CHECK("VDOpen()");
149
150 VDDumpImages(pVD);
151
152 VDClose(pVD, true);
153 RTFILE File;
154 rc = RTFileOpen(&File, pszFilename, RTFILE_O_READ);
155 if (VBOX_SUCCESS(rc))
156 {
157 RTFileClose(File);
158 return VERR_INTERNAL_ERROR;
159 }
160
161 VDDestroy(pVD);
162#undef CHECK
163 return 0;
164}
165
166
167#undef RTDECL
168#define RTDECL(x) static x
169
170/* Start of IPRT code */
171
172/**
173 * The following code is based on the work of George Marsaglia
174 * taken from
175 * http://groups.google.ws/group/comp.sys.sun.admin/msg/7c667186f6cbf354
176 * and
177 * http://groups.google.ws/group/comp.lang.c/msg/0e170777c6e79e8d
178 */
179
180/*
181A C version of a very very good 64-bit RNG is given below.
182You should be able to adapt it to your particular needs.
183
184It is based on the complimentary-multiple-with-carry
185sequence
186 x(n)=a*x(n-4)+carry mod 2^64-1,
187which works as follows:
188Assume a certain multiplier 'a' and a base 'b'.
189Given a current x value and a current carry 'c',
190form: t=a*x+c
191Then the new carry is c=floor(t/b)
192and the new x value is x = b-1-(t mod b).
193
194
195Ordinarily, for 32-bit mwc or cmwc sequences, the
196value t=a*x+c can be formed in 64 bits, then the new c
197is the top and the new x the bottom 32 bits (with a little
198fiddling when b=2^32-1 and cmwc rather than mwc.)
199
200
201To generate 64-bit x's, it is difficult to form
202t=a*x+c in 128 bits then get the new c and new x
203from the the top and bottom halves.
204But if 'a' has a special form, for example,
205a=2^62+2^47+2 and b=2^64-1, then the new c and
206the new x can be formed with shifts, tests and +/-'s,
207again with a little fiddling because b=2^64-1 rather
208than 2^64. (The latter is not an optimal choice because,
209being a square, it cannot be a primitive root of the
210prime a*b^k+1, where 'k' is the 'lag':
211 x(n)=a*x(n-k)+carry mod b.)
212But the multiplier a=2^62+2^47+2 makes a*b^4+1 a prime for
213which b=2^64-1 is a primitive root, and getting the new x and
214new c can be done with arithmetic on integers the size of x.
215*/
216
217struct RndCtx
218{
219 uint64_t x;
220 uint64_t y;
221 uint64_t z;
222 uint64_t w;
223 uint64_t c;
224 uint32_t u32x;
225 uint32_t u32y;
226};
227typedef struct RndCtx RNDCTX;
228typedef RNDCTX *PRNDCTX;
229
230/**
231 * Initialize seeds.
232 *
233 * @remarks You should choose ANY 4 random 64-bit
234 * seeds x,y,z,w < 2^64-1 and a random seed c in
235 * 0<= c < a = 2^62+2^47+2.
236 * There are P=(2^62+2^46+2)*(2^64-1)^4 > 2^318 possible choices
237 * for seeds, the period of the RNG.
238 */
239RTDECL(int) RTPRandInit(PRNDCTX pCtx, uint32_t u32Seed)
240{
241 if (u32Seed == 0)
242 u32Seed = (uint32_t)(ASMReadTSC() >> 8);
243 /* Zero is not a good seed. */
244 if (u32Seed == 0)
245 u32Seed = 362436069;
246 pCtx->x = u32Seed;
247 pCtx->y = 17280675555674358941ULL;
248 pCtx->z = 6376492577913983186ULL;
249 pCtx->w = 9064188857900113776ULL;
250 pCtx->c = 123456789;
251 pCtx->u32x = 2282008;
252 pCtx->u32y = u32Seed;
253 return VINF_SUCCESS;
254}
255
256RTDECL(uint32_t) RTPRandGetSeedInfo(PRNDCTX pCtx)
257{
258 return pCtx->u32y;
259}
260
261/**
262 * Generate a 64-bit unsigned random number.
263 *
264 * @returns The pseudo random number.
265 */
266RTDECL(uint64_t) RTPRandU64(PRNDCTX pCtx)
267{
268 uint64_t t;
269 t = (pCtx->x<<47) + (pCtx->x<<62) + (pCtx->x<<1);
270 t += pCtx->c; t+= (t < pCtx->c);
271 pCtx->c = (t<pCtx->c) + (pCtx->x>>17) + (pCtx->x>>2) + (pCtx->x>>63);
272 pCtx->x = pCtx->y; pCtx->y = pCtx->z ; pCtx->z = pCtx->w;
273 return (pCtx->w = ~(t + pCtx->c)-1);
274}
275
276/**
277 * Generate a 64-bit unsigned pseudo random number in the set
278 * [u64First..u64Last].
279 *
280 * @returns The pseudo random number.
281 * @param u64First First number in the set.
282 * @param u64Last Last number in the set.
283 */
284RTDECL(uint64_t) RTPRandU64Ex(PRNDCTX pCtx, uint64_t u64First, uint64_t u64Last)
285{
286 if (u64First == 0 && u64Last == UINT64_MAX)
287 return RTPRandU64(pCtx);
288
289 uint64_t u64Tmp;
290 uint64_t u64Range = u64Last - u64First + 1;
291 uint64_t u64Scale = UINT64_MAX / u64Range;
292
293 do
294 {
295 u64Tmp = RTPRandU64(pCtx) / u64Scale;
296 } while (u64Tmp >= u64Range);
297 return u64First + u64Tmp;
298}
299
300/**
301 * Generate a 32-bit unsigned random number.
302 *
303 * @returns The pseudo random number.
304 */
305RTDECL(uint32_t) RTPRandU32(PRNDCTX pCtx)
306{
307 return ( pCtx->u32x = 69069 * pCtx->u32x + 123,
308 pCtx->u32y ^= pCtx->u32y<<13,
309 pCtx->u32y ^= pCtx->u32y>>17,
310 pCtx->u32y ^= pCtx->u32y<<5,
311 pCtx->u32x + pCtx->u32y );
312}
313
314/**
315 * Generate a 32-bit unsigned pseudo random number in the set
316 * [u32First..u32Last].
317 *
318 * @returns The pseudo random number.
319 * @param u32First First number in the set.
320 * @param u32Last Last number in the set.
321 */
322RTDECL(uint32_t) RTPRandU32Ex(PRNDCTX pCtx, uint32_t u32First, uint32_t u32Last)
323{
324 if (u32First == 0 && u32Last == UINT32_MAX)
325 return RTPRandU32(pCtx);
326
327 uint32_t u32Tmp;
328 uint32_t u32Range = u32Last - u32First + 1;
329 uint32_t u32Scale = UINT32_MAX / u32Range;
330
331 do
332 {
333 u32Tmp = RTPRandU32(pCtx) / u32Scale;
334 } while (u32Tmp >= u32Range);
335 return u32First + u32Tmp;
336}
337
338/* End of IPRT code */
339
340struct Segment
341{
342 uint64_t u64Offset;
343 uint32_t u32Length;
344 uint32_t u8Value;
345};
346typedef struct Segment *PSEGMENT;
347
348static void initializeRandomGenerator(PRNDCTX pCtx, uint32_t u32Seed)
349{
350 int rc = RTPRandInit(pCtx, u32Seed);
351 if (VBOX_FAILURE(rc))
352 RTPrintf("ERROR: Failed to initialize random generator. RC=%Vrc\n", rc);
353 else
354 {
355 RTPrintf("INFO: Random generator seed used: %x\n", RTPRandGetSeedInfo(pCtx));
356 RTLogPrintf("INFO: Random generator seed used: %x\n", RTPRandGetSeedInfo(pCtx));
357 }
358}
359
360static int compareSegments(const void *left, const void *right)
361{
362 /* Note that no duplicates are allowed in the array being sorted. */
363 return ((PSEGMENT)left)->u64Offset < ((PSEGMENT)right)->u64Offset ? -1 : 1;
364}
365
366static void generateRandomSegments(PRNDCTX pCtx, PSEGMENT pSegment, uint32_t nSegments, uint32_t u32MaxSegmentSize, uint64_t u64DiskSize, uint32_t u32SectorSize, uint8_t u8ValueLow, uint8_t u8ValueHigh)
367{
368 uint32_t i;
369 /* Generate segment offsets. */
370 for (i = 0; i < nSegments; i++)
371 {
372 bool fDuplicateFound;
373 do
374 {
375 pSegment[i].u64Offset = RTPRandU64Ex(pCtx, 0, u64DiskSize / u32SectorSize - 1) * u32SectorSize;
376 fDuplicateFound = false;
377 for (uint32_t j = 0; j < i; j++)
378 if (pSegment[i].u64Offset == pSegment[j].u64Offset)
379 {
380 fDuplicateFound = true;
381 break;
382 }
383 } while (fDuplicateFound);
384 }
385 /* Sort in offset-ascending order. */
386 qsort(pSegment, nSegments, sizeof(*pSegment), compareSegments);
387 /* Put a sentinel at the end. */
388 pSegment[nSegments].u64Offset = u64DiskSize;
389 pSegment[nSegments].u32Length = 0;
390 /* Generate segment lengths and values. */
391 for (i = 0; i < nSegments; i++)
392 {
393 pSegment[i].u32Length = RTPRandU32Ex(pCtx, 1, RT_MIN(pSegment[i+1].u64Offset - pSegment[i].u64Offset,
394 u32MaxSegmentSize) / u32SectorSize) * u32SectorSize;
395 Assert(pSegment[i].u32Length <= u32MaxSegmentSize);
396 pSegment[i].u8Value = RTPRandU32Ex(pCtx, (uint32_t)u8ValueLow, (uint32_t)u8ValueHigh);
397 }
398}
399
400static void mergeSegments(PSEGMENT pBaseSegment, PSEGMENT pDiffSegment, PSEGMENT pMergeSegment, uint32_t u32MaxLength)
401{
402 while (pBaseSegment->u32Length > 0 || pDiffSegment->u32Length > 0)
403 {
404 if (pBaseSegment->u64Offset < pDiffSegment->u64Offset)
405 {
406 *pMergeSegment = *pBaseSegment;
407 if (pMergeSegment->u64Offset + pMergeSegment->u32Length <= pDiffSegment->u64Offset)
408 pBaseSegment++;
409 else
410 {
411 pMergeSegment->u32Length = pDiffSegment->u64Offset - pMergeSegment->u64Offset;
412 Assert(pMergeSegment->u32Length <= u32MaxLength);
413 if (pBaseSegment->u64Offset + pBaseSegment->u32Length >
414 pDiffSegment->u64Offset + pDiffSegment->u32Length)
415 {
416 pBaseSegment->u32Length -= pDiffSegment->u64Offset + pDiffSegment->u32Length - pBaseSegment->u64Offset;
417 Assert(pBaseSegment->u32Length <= u32MaxLength);
418 pBaseSegment->u64Offset = pDiffSegment->u64Offset + pDiffSegment->u32Length;
419 }
420 else
421 pBaseSegment++;
422 }
423 pMergeSegment++;
424 }
425 else
426 {
427 *pMergeSegment = *pDiffSegment;
428 if (pMergeSegment->u64Offset + pMergeSegment->u32Length <= pBaseSegment->u64Offset)
429 {
430 pDiffSegment++;
431 pMergeSegment++;
432 }
433 else
434 {
435 if (pBaseSegment->u64Offset + pBaseSegment->u32Length > pDiffSegment->u64Offset + pDiffSegment->u32Length)
436 {
437 pBaseSegment->u32Length -= pDiffSegment->u64Offset + pDiffSegment->u32Length - pBaseSegment->u64Offset;
438 Assert(pBaseSegment->u32Length <= u32MaxLength);
439 pBaseSegment->u64Offset = pDiffSegment->u64Offset + pDiffSegment->u32Length;
440 pDiffSegment++;
441 pMergeSegment++;
442 }
443 else
444 pBaseSegment++;
445 }
446 }
447 }
448}
449
450static void writeSegmentsToDisk(PVBOXHDD pVD, void *pvBuf, PSEGMENT pSegment)
451{
452 while (pSegment->u32Length)
453 {
454 //memset((uint8_t*)pvBuf + pSegment->u64Offset, pSegment->u8Value, pSegment->u32Length);
455 memset(pvBuf, pSegment->u8Value, pSegment->u32Length);
456 VDWrite(pVD, pSegment->u64Offset, pvBuf, pSegment->u32Length);
457 pSegment++;
458 }
459}
460
461static int readAndCompareSegments(PVBOXHDD pVD, void *pvBuf, PSEGMENT pSegment)
462{
463 while (pSegment->u32Length)
464 {
465 int rc = VDRead(pVD, pSegment->u64Offset, pvBuf, pSegment->u32Length);
466 if (VBOX_FAILURE(rc))
467 {
468 RTPrintf("ERROR: Failed to read from virtual disk\n");
469 return rc;
470 }
471 else
472 {
473 for (unsigned i = 0; i < pSegment->u32Length; i++)
474 if (((uint8_t*)pvBuf)[i] != pSegment->u8Value)
475 {
476 RTPrintf("ERROR: Segment at %Lx of %x bytes is corrupt at offset %x (found %x instead of %x)\n",
477 pSegment->u64Offset, pSegment->u32Length, i, ((uint8_t*)pvBuf)[i],
478 pSegment->u8Value);
479 RTLogPrintf("ERROR: Segment at %Lx of %x bytes is corrupt at offset %x (found %x instead of %x)\n",
480 pSegment->u64Offset, pSegment->u32Length, i, ((uint8_t*)pvBuf)[i],
481 pSegment->u8Value);
482 return VERR_INTERNAL_ERROR;
483 }
484 }
485 pSegment++;
486 }
487
488 return VINF_SUCCESS;
489}
490
491static int tstVDOpenCreateWriteMerge(const char *pszBackend,
492 const char *pszBaseFilename,
493 const char *pszDiffFilename,
494 uint32_t u32Seed)
495{
496 int rc;
497 PVBOXHDD pVD = NULL;
498 char *pszFormat;
499 PDMMEDIAGEOMETRY PCHS = { 0, 0, 0 };
500 PDMMEDIAGEOMETRY LCHS = { 0, 0, 0 };
501 uint64_t u64DiskSize = 1000 * _1M;
502 uint32_t u32SectorSize = 512;
503 PVDINTERFACE pVDIfs = NULL;
504 VDINTERFACE VDIError;
505 VDINTERFACEERROR VDIErrorCallbacks;
506
507#define CHECK(str) \
508 do \
509 { \
510 RTPrintf("%s rc=%Vrc\n", str, rc); \
511 if (VBOX_FAILURE(rc)) \
512 { \
513 if (pvBuf) \
514 RTMemFree(pvBuf); \
515 VDDestroy(pVD); \
516 return rc; \
517 } \
518 } while (0)
519
520 void *pvBuf = RTMemAlloc(_1M);
521
522 /* Create error interface. */
523 VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
524 VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
525 VDIErrorCallbacks.pfnError = tstVDError;
526
527 rc = VDInterfaceAdd(&VDIError, "tstVD_Error", VDINTERFACETYPE_ERROR, &VDIErrorCallbacks,
528 NULL, &pVDIfs);
529 AssertRC(rc);
530
531
532 rc = VDCreate(&VDIError, &pVD);
533 CHECK("VDCreate()");
534
535 RTFILE File;
536 rc = RTFileOpen(&File, pszBaseFilename, RTFILE_O_READ);
537 if (VBOX_SUCCESS(rc))
538 {
539 RTFileClose(File);
540 rc = VDGetFormat(pszBaseFilename, &pszFormat);
541 RTPrintf("VDGetFormat() pszFormat=%s rc=%Vrc\n", pszFormat, rc);
542 if (VBOX_SUCCESS(rc) && strcmp(pszFormat, pszBackend))
543 {
544 rc = VERR_GENERAL_FAILURE;
545 RTPrintf("VDGetFormat() returned incorrect backend name\n");
546 }
547 RTStrFree(pszFormat);
548 CHECK("VDGetFormat()");
549
550 rc = VDOpen(pVD, pszBackend, pszBaseFilename, VD_OPEN_FLAGS_NORMAL,
551 NULL);
552 CHECK("VDOpen()");
553 }
554 else
555 {
556 rc = VDCreateBase(pVD, pszBackend, pszBaseFilename,
557 VD_IMAGE_TYPE_NORMAL, u64DiskSize,
558 VD_IMAGE_FLAGS_NONE, "Test image",
559 &PCHS, &LCHS, NULL, VD_OPEN_FLAGS_NORMAL,
560 NULL, NULL);
561 CHECK("VDCreateBase()");
562 }
563
564 int nSegments = 100;
565 /* Allocate one extra element for a sentinel. */
566 PSEGMENT paBaseSegments = (PSEGMENT)RTMemAllocZ(sizeof(struct Segment) * (nSegments + 1));
567 PSEGMENT paDiffSegments = (PSEGMENT)RTMemAllocZ(sizeof(struct Segment) * (nSegments + 1));
568 PSEGMENT paMergeSegments = (PSEGMENT)RTMemAllocZ(sizeof(struct Segment) * (nSegments + 1) * 3);
569
570 RNDCTX ctx;
571 initializeRandomGenerator(&ctx, u32Seed);
572 generateRandomSegments(&ctx, paBaseSegments, nSegments, _1M, u64DiskSize, u32SectorSize, 0u, 127u);
573 generateRandomSegments(&ctx, paDiffSegments, nSegments, _1M, u64DiskSize, u32SectorSize, 128u, 255u);
574
575 /*PSEGMENT pSegment;
576 RTPrintf("Base segments:\n");
577 for (pSegment = paBaseSegments; pSegment->u32Length; pSegment++)
578 RTPrintf("off: %08Lx len: %05x val: %02x\n", pSegment->u64Offset, pSegment->u32Length, pSegment->u8Value);*/
579 writeSegmentsToDisk(pVD, pvBuf, paBaseSegments);
580
581 rc = VDCreateDiff(pVD, pszBackend, pszDiffFilename,
582 VD_IMAGE_FLAGS_NONE, "Test diff image", NULL,
583 VD_OPEN_FLAGS_NORMAL, NULL, NULL);
584 CHECK("VDCreateDiff()");
585
586 /*RTPrintf("\nDiff segments:\n");
587 for (pSegment = paDiffSegments; pSegment->u32Length; pSegment++)
588 RTPrintf("off: %08Lx len: %05x val: %02x\n", pSegment->u64Offset, pSegment->u32Length, pSegment->u8Value);*/
589 writeSegmentsToDisk(pVD, pvBuf, paDiffSegments);
590
591 VDDumpImages(pVD);
592
593 RTPrintf("Merging diff into base..\n");
594 rc = VDMerge(pVD, VD_LAST_IMAGE, 0, NULL);
595 CHECK("VDMerge()");
596
597 mergeSegments(paBaseSegments, paDiffSegments, paMergeSegments, _1M);
598 /*RTPrintf("\nMerged segments:\n");
599 for (pSegment = paMergeSegments; pSegment->u32Length; pSegment++)
600 RTPrintf("off: %08Lx len: %05x val: %02x\n", pSegment->u64Offset, pSegment->u32Length, pSegment->u8Value);*/
601 rc = readAndCompareSegments(pVD, pvBuf, paMergeSegments);
602 CHECK("readAndCompareSegments()");
603
604 RTMemFree(paMergeSegments);
605 RTMemFree(paDiffSegments);
606 RTMemFree(paBaseSegments);
607
608 VDDumpImages(pVD);
609
610 VDDestroy(pVD);
611 if (pvBuf)
612 RTMemFree(pvBuf);
613#undef CHECK
614 return 0;
615}
616
617static int tstVDCreateWriteOpenRead(const char *pszBackend,
618 const char *pszFilename,
619 uint32_t u32Seed)
620{
621 int rc;
622 PVBOXHDD pVD = NULL;
623 PDMMEDIAGEOMETRY PCHS = { 0, 0, 0 };
624 PDMMEDIAGEOMETRY LCHS = { 0, 0, 0 };
625 uint64_t u64DiskSize = 1000 * _1M;
626 uint32_t u32SectorSize = 512;
627 PVDINTERFACE pVDIfs;
628 VDINTERFACE VDIError;
629 VDINTERFACEERROR VDIErrorCallbacks;
630
631#define CHECK(str) \
632 do \
633 { \
634 RTPrintf("%s rc=%Vrc\n", str, rc); \
635 if (VBOX_FAILURE(rc)) \
636 { \
637 if (pvBuf) \
638 RTMemFree(pvBuf); \
639 VDDestroy(pVD); \
640 return rc; \
641 } \
642 } while (0)
643
644 void *pvBuf = RTMemAlloc(_1M);
645
646 /* Create error interface. */
647 VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
648 VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
649 VDIErrorCallbacks.pfnError = tstVDError;
650
651 rc = VDInterfaceAdd(&VDIError, "tstVD_Error", VDINTERFACETYPE_ERROR, &VDIErrorCallbacks,
652 NULL, &pVDIfs);
653 AssertRC(rc);
654
655
656 rc = VDCreate(&VDIError, &pVD);
657 CHECK("VDCreate()");
658
659 RTFILE File;
660 rc = RTFileOpen(&File, pszFilename, RTFILE_O_READ);
661 if (VBOX_SUCCESS(rc))
662 {
663 RTFileClose(File);
664 RTFileDelete(pszFilename);
665 }
666
667 rc = VDCreateBase(pVD, pszBackend, pszFilename,
668 VD_IMAGE_TYPE_NORMAL, u64DiskSize,
669 VD_IMAGE_FLAGS_NONE, "Test image",
670 &PCHS, &LCHS, NULL, VD_OPEN_FLAGS_NORMAL,
671 NULL, NULL);
672 CHECK("VDCreateBase()");
673
674 int nSegments = 100;
675 /* Allocate one extra element for a sentinel. */
676 PSEGMENT paSegments = (PSEGMENT)RTMemAllocZ(sizeof(struct Segment) * (nSegments + 1));
677
678 RNDCTX ctx;
679 initializeRandomGenerator(&ctx, u32Seed);
680 generateRandomSegments(&ctx, paSegments, nSegments, _1M, u64DiskSize, u32SectorSize, 0u, 127u);
681 /*for (PSEGMENT pSegment = paSegments; pSegment->u32Length; pSegment++)
682 RTPrintf("off: %08Lx len: %05x val: %02x\n", pSegment->u64Offset, pSegment->u32Length, pSegment->u8Value);*/
683
684 writeSegmentsToDisk(pVD, pvBuf, paSegments);
685
686 VDCloseAll(pVD);
687
688 rc = VDOpen(pVD, pszBackend, pszFilename, VD_OPEN_FLAGS_NORMAL, NULL);
689 CHECK("VDOpen()");
690 rc = readAndCompareSegments(pVD, pvBuf, paSegments);
691 CHECK("readAndCompareSegments()");
692
693 RTMemFree(paSegments);
694
695 VDDestroy(pVD);
696 if (pvBuf)
697 RTMemFree(pvBuf);
698#undef CHECK
699 return 0;
700}
701
702static int tstVmdkRename(const char *src, const char *dst)
703{
704 int rc;
705 PVBOXHDD pVD = NULL;
706 PVDINTERFACE pVDIfs = NULL;
707 VDINTERFACE VDIError;
708 VDINTERFACEERROR VDIErrorCallbacks;
709
710#define CHECK(str) \
711 do \
712 { \
713 RTPrintf("%s rc=%Vrc\n", str, rc); \
714 if (VBOX_FAILURE(rc)) \
715 { \
716 VDDestroy(pVD); \
717 return rc; \
718 } \
719 } while (0)
720
721 /* Create error interface. */
722 VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
723 VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
724 VDIErrorCallbacks.pfnError = tstVDError;
725
726 rc = VDInterfaceAdd(&VDIError, "tstVD_Error", VDINTERFACETYPE_ERROR, &VDIErrorCallbacks,
727 NULL, &pVDIfs);
728 AssertRC(rc);
729
730 rc = VDCreate(&VDIError, &pVD);
731 CHECK("VDCreate()");
732
733 rc = VDOpen(pVD, "VMDK", src, VD_OPEN_FLAGS_NORMAL, NULL);
734 CHECK("VDOpen()");
735 rc = VDCopy(pVD, 0, pVD, "VMDK", dst, true, 0, NULL, NULL, NULL);
736 CHECK("VDCopy()");
737
738 VDDestroy(pVD);
739#undef CHECK
740 return 0;
741}
742
743static int tstVmdkCreateRenameOpen(const char *src, const char *dst,
744 uint64_t cbSize, VDIMAGETYPE enmType,
745 unsigned uFlags)
746{
747 int rc = tstVDCreateDelete("VMDK", src, cbSize, enmType, uFlags, false);
748 if (VBOX_FAILURE(rc))
749 return rc;
750
751 rc = tstVmdkRename(src, dst);
752 if (VBOX_FAILURE(rc))
753 return rc;
754
755 PVBOXHDD pVD = NULL;
756 PVDINTERFACE pVDIfs = NULL;
757 VDINTERFACE VDIError;
758 VDINTERFACEERROR VDIErrorCallbacks;
759
760#define CHECK(str) \
761 do \
762 { \
763 RTPrintf("%s rc=%Vrc\n", str, rc); \
764 if (VBOX_FAILURE(rc)) \
765 { \
766 VDCloseAll(pVD); \
767 return rc; \
768 } \
769 } while (0)
770
771 /* Create error interface. */
772 VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
773 VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
774 VDIErrorCallbacks.pfnError = tstVDError;
775
776 rc = VDInterfaceAdd(&VDIError, "tstVD_Error", VDINTERFACETYPE_ERROR, &VDIErrorCallbacks,
777 NULL, &pVDIfs);
778 AssertRC(rc);
779
780 rc = VDCreate(&VDIError, &pVD);
781 CHECK("VDCreate()");
782
783 rc = VDOpen(pVD, "VMDK", dst, VD_OPEN_FLAGS_NORMAL, NULL);
784 CHECK("VDOpen()");
785
786 VDClose(pVD, true);
787 CHECK("VDClose()");
788 VDDestroy(pVD);
789#undef CHECK
790 return rc;
791}
792
793#if defined(RT_OS_OS2) || defined(RT_OS_WINDOWS)
794#define DST_PATH "tmp\\tmpVDRename.vmdk"
795#else
796#define DST_PATH "tmp/tmpVDRename.vmdk"
797#endif
798
799static void tstVmdk()
800{
801 int rc = tstVmdkCreateRenameOpen("tmpVDCreate.vmdk", "tmpVDRename.vmdk", _4G,
802 VD_IMAGE_TYPE_NORMAL, VD_IMAGE_FLAGS_NONE);
803 if (VBOX_FAILURE(rc))
804 {
805 RTPrintf("tstVD: VMDK rename (single extent, embedded descriptor, same dir) test failed! rc=%Vrc\n", rc);
806 g_cErrors++;
807 }
808 rc = tstVmdkCreateRenameOpen("tmpVDCreate.vmdk", "tmpVDRename.vmdk", _4G,
809 VD_IMAGE_TYPE_NORMAL, VD_VMDK_IMAGE_FLAGS_SPLIT_2G);
810 if (VBOX_FAILURE(rc))
811 {
812 RTPrintf("tstVD: VMDK rename (multiple extent, separate descriptor, same dir) test failed! rc=%Vrc\n", rc);
813 g_cErrors++;
814 }
815 rc = tstVmdkCreateRenameOpen("tmpVDCreate.vmdk", DST_PATH, _4G,
816 VD_IMAGE_TYPE_NORMAL, VD_IMAGE_FLAGS_NONE);
817 if (VBOX_FAILURE(rc))
818 {
819 RTPrintf("tstVD: VMDK rename (single extent, embedded descriptor, another dir) test failed! rc=%Vrc\n", rc);
820 g_cErrors++;
821 }
822 rc = tstVmdkCreateRenameOpen("tmpVDCreate.vmdk", DST_PATH, _4G,
823 VD_IMAGE_TYPE_NORMAL, VD_VMDK_IMAGE_FLAGS_SPLIT_2G);
824 if (VBOX_FAILURE(rc))
825 {
826 RTPrintf("tstVD: VMDK rename (multiple extent, separate descriptor, another dir) test failed! rc=%Vrc\n", rc);
827 g_cErrors++;
828 }
829
830 RTFILE File;
831 rc = RTFileOpen(&File, DST_PATH, RTFILE_O_CREATE | RTFILE_O_WRITE);
832 if (VBOX_SUCCESS(rc))
833 RTFileClose(File);
834
835 rc = tstVmdkCreateRenameOpen("tmpVDCreate.vmdk", DST_PATH, _4G,
836 VD_IMAGE_TYPE_NORMAL, VD_VMDK_IMAGE_FLAGS_SPLIT_2G);
837 if (VBOX_SUCCESS(rc))
838 {
839 RTPrintf("tstVD: VMDK rename (multiple extent, separate descriptor, another dir, already exists) test failed!\n");
840 g_cErrors++;
841 }
842 RTFileDelete(DST_PATH);
843 RTFileDelete("tmpVDCreate.vmdk");
844 RTFileDelete("tmpVDCreate-s001.vmdk");
845 RTFileDelete("tmpVDCreate-s002.vmdk");
846 RTFileDelete("tmpVDCreate-s003.vmdk");
847}
848
849int main(int argc, char *argv[])
850{
851 int rc;
852
853 uint32_t u32Seed = 0; // Means choose random
854
855 if (argc > 1)
856 if (sscanf(argv[1], "%x", &u32Seed) != 1)
857 {
858 RTPrintf("ERROR: Invalid parameter %s. Valid usage is %s <32-bit seed>.\n",
859 argv[1], argv[0]);
860 return 1;
861 }
862
863 RTR3Init();
864 RTPrintf("tstVD: TESTING...\n");
865
866 /*
867 * Clean up potential leftovers from previous unsuccessful runs.
868 */
869 RTFileDelete("tmpVDCreate.vdi");
870 RTFileDelete("tmpVDCreate.vmdk");
871 RTFileDelete("tmpVDCreate.vhd");
872 RTFileDelete("tmpVDBase.vdi");
873 RTFileDelete("tmpVDDiff.vdi");
874 RTFileDelete("tmpVDBase.vmdk");
875 RTFileDelete("tmpVDDiff.vmdk");
876 RTFileDelete("tmpVDBase.vhd");
877 RTFileDelete("tmpVDDiff.vhd");
878 RTFileDelete("tmpVDCreate-s001.vmdk");
879 RTFileDelete("tmpVDCreate-s002.vmdk");
880 RTFileDelete("tmpVDCreate-s003.vmdk");
881 RTFileDelete("tmpVDRename.vmdk");
882 RTFileDelete("tmpVDRename-s001.vmdk");
883 RTFileDelete("tmpVDRename-s002.vmdk");
884 RTFileDelete("tmpVDRename-s003.vmdk");
885 RTFileDelete("tmp/tmpVDRename.vmdk");
886 RTFileDelete("tmp/tmpVDRename-s001.vmdk");
887 RTFileDelete("tmp/tmpVDRename-s002.vmdk");
888 RTFileDelete("tmp/tmpVDRename-s003.vmdk");
889
890 if (!RTDirExists("tmp"))
891 {
892 rc = RTDirCreate("tmp", RTFS_UNIX_IRWXU);
893 if (VBOX_FAILURE(rc))
894 {
895 RTPrintf("tstVD: Failed to create 'tmp' directory! rc=%Vrc\n", rc);
896 g_cErrors++;
897 }
898 }
899
900#ifdef VMDK_TEST
901 rc = tstVDCreateDelete("VMDK", "tmpVDCreate.vmdk", 2 * _4G,
902 VD_IMAGE_TYPE_NORMAL, VD_IMAGE_FLAGS_NONE,
903 true);
904 if (VBOX_FAILURE(rc))
905 {
906 RTPrintf("tstVD: dynamic VMDK create test failed! rc=%Vrc\n", rc);
907 g_cErrors++;
908 }
909 rc = tstVDCreateDelete("VMDK", "tmpVDCreate.vmdk", 2 * _4G,
910 VD_IMAGE_TYPE_NORMAL, VD_IMAGE_FLAGS_NONE,
911 false);
912 if (VBOX_FAILURE(rc))
913 {
914 RTPrintf("tstVD: dynamic VMDK create test failed! rc=%Vrc\n", rc);
915 g_cErrors++;
916 }
917 rc = tstVDOpenDelete("VMDK", "tmpVDCreate.vmdk");
918 if (VBOX_FAILURE(rc))
919 {
920 RTPrintf("tstVD: VMDK delete test failed! rc=%Vrc\n", rc);
921 g_cErrors++;
922 }
923
924 tstVmdk();
925#endif /* VMDK_TEST */
926#ifdef VDI_TEST
927 rc = tstVDCreateDelete("VDI", "tmpVDCreate.vdi", 2 * _4G,
928 VD_IMAGE_TYPE_NORMAL, VD_IMAGE_FLAGS_NONE,
929 true);
930 if (VBOX_FAILURE(rc))
931 {
932 RTPrintf("tstVD: dynamic VDI create test failed! rc=%Vrc\n", rc);
933 g_cErrors++;
934 }
935 rc = tstVDCreateDelete("VDI", "tmpVDCreate.vdi", 2 * _4G,
936 VD_IMAGE_TYPE_FIXED, VD_IMAGE_FLAGS_NONE,
937 true);
938 if (VBOX_FAILURE(rc))
939 {
940 RTPrintf("tstVD: fixed VDI create test failed! rc=%Vrc\n", rc);
941 g_cErrors++;
942 }
943#endif /* VDI_TEST */
944#ifdef VMDK_TEST
945 rc = tstVDCreateDelete("VMDK", "tmpVDCreate.vmdk", 2 * _4G,
946 VD_IMAGE_TYPE_NORMAL, VD_IMAGE_FLAGS_NONE,
947 true);
948 if (VBOX_FAILURE(rc))
949 {
950 RTPrintf("tstVD: dynamic VMDK create test failed! rc=%Vrc\n", rc);
951 g_cErrors++;
952 }
953 rc = tstVDCreateDelete("VMDK", "tmpVDCreate.vmdk", 2 * _4G,
954 VD_IMAGE_TYPE_NORMAL, VD_VMDK_IMAGE_FLAGS_SPLIT_2G,
955 true);
956 if (VBOX_FAILURE(rc))
957 {
958 RTPrintf("tstVD: dynamic split VMDK create test failed! rc=%Vrc\n", rc);
959 g_cErrors++;
960 }
961 rc = tstVDCreateDelete("VMDK", "tmpVDCreate.vmdk", 2 * _4G,
962 VD_IMAGE_TYPE_FIXED, VD_IMAGE_FLAGS_NONE,
963 true);
964 if (VBOX_FAILURE(rc))
965 {
966 RTPrintf("tstVD: fixed VMDK create test failed! rc=%Vrc\n", rc);
967 g_cErrors++;
968 }
969 rc = tstVDCreateDelete("VMDK", "tmpVDCreate.vmdk", 2 * _4G,
970 VD_IMAGE_TYPE_FIXED, VD_VMDK_IMAGE_FLAGS_SPLIT_2G,
971 true);
972 if (VBOX_FAILURE(rc))
973 {
974 RTPrintf("tstVD: fixed split VMDK create test failed! rc=%Vrc\n", rc);
975 g_cErrors++;
976 }
977#endif /* VMDK_TEST */
978#ifdef VHD_TEST
979 rc = tstVDCreateDelete("VHD", "tmpVDCreate.vhd", 2 * _4G,
980 VD_IMAGE_TYPE_NORMAL, VD_IMAGE_FLAGS_NONE,
981 true);
982 if (VBOX_FAILURE(rc))
983 {
984 RTPrintf("tstVD: dynamic VHD create test failed! rc=%Vrc\n", rc);
985 g_cErrors++;
986 }
987 rc = tstVDCreateDelete("VHD", "tmpVDCreate.vhd", 2 * _4G,
988 VD_IMAGE_TYPE_FIXED, VD_IMAGE_FLAGS_NONE,
989 true);
990 if (VBOX_FAILURE(rc))
991 {
992 RTPrintf("tstVD: fixed VHD create test failed! rc=%Vrc\n", rc);
993 g_cErrors++;
994 }
995#endif /* VHD_TEST */
996#ifdef VDI_TEST
997 rc = tstVDOpenCreateWriteMerge("VDI", "tmpVDBase.vdi", "tmpVDDiff.vdi", u32Seed);
998 if (VBOX_FAILURE(rc))
999 {
1000 RTPrintf("tstVD: VDI test failed (new image)! rc=%Vrc\n", rc);
1001 g_cErrors++;
1002 }
1003 rc = tstVDOpenCreateWriteMerge("VDI", "tmpVDBase.vdi", "tmpVDDiff.vdi", u32Seed);
1004 if (VBOX_FAILURE(rc))
1005 {
1006 RTPrintf("tstVD: VDI test failed (existing image)! rc=%Vrc\n", rc);
1007 g_cErrors++;
1008 }
1009#endif /* VDI_TEST */
1010#ifdef VMDK_TEST
1011 rc = tstVDOpenCreateWriteMerge("VMDK", "tmpVDBase.vmdk", "tmpVDDiff.vmdk", u32Seed);
1012 if (VBOX_FAILURE(rc))
1013 {
1014 RTPrintf("tstVD: VMDK test failed (new image)! rc=%Vrc\n", rc);
1015 g_cErrors++;
1016 }
1017 rc = tstVDOpenCreateWriteMerge("VMDK", "tmpVDBase.vmdk", "tmpVDDiff.vmdk", u32Seed);
1018 if (VBOX_FAILURE(rc))
1019 {
1020 RTPrintf("tstVD: VMDK test failed (existing image)! rc=%Vrc\n", rc);
1021 g_cErrors++;
1022 }
1023#endif /* VMDK_TEST */
1024#ifdef VHD_TEST
1025 rc = tstVDCreateWriteOpenRead("VHD", "tmpVDCreate.vhd", u32Seed);
1026 if (VBOX_FAILURE(rc))
1027 {
1028 RTPrintf("tstVD: VHD test failed (creating image)! rc=%Vrc\n", rc);
1029 g_cErrors++;
1030 }
1031
1032 rc = tstVDOpenCreateWriteMerge("VHD", "tmpVDBase.vhd", "tmpVDDiff.vhd", u32Seed);
1033 if (VBOX_FAILURE(rc))
1034 {
1035 RTPrintf("tstVD: VHD test failed (existing image)! rc=%Vrc\n", rc);
1036 g_cErrors++;
1037 }
1038#endif /* VHD_TEST */
1039
1040 /*
1041 * Clean up any leftovers.
1042 */
1043 RTFileDelete("tmpVDCreate.vdi");
1044 RTFileDelete("tmpVDCreate.vmdk");
1045 RTFileDelete("tmpVDCreate.vhd");
1046 RTFileDelete("tmpVDBase.vdi");
1047 RTFileDelete("tmpVDDiff.vdi");
1048 RTFileDelete("tmpVDBase.vmdk");
1049 RTFileDelete("tmpVDDiff.vmdk");
1050 RTFileDelete("tmpVDBase.vhd");
1051 RTFileDelete("tmpVDDiff.vhd");
1052 RTFileDelete("tmpVDCreate-s001.vmdk");
1053 RTFileDelete("tmpVDCreate-s002.vmdk");
1054 RTFileDelete("tmpVDCreate-s003.vmdk");
1055 RTFileDelete("tmpVDRename.vmdk");
1056 RTFileDelete("tmpVDRename-s001.vmdk");
1057 RTFileDelete("tmpVDRename-s002.vmdk");
1058 RTFileDelete("tmpVDRename-s003.vmdk");
1059
1060 /*
1061 * Summary
1062 */
1063 if (!g_cErrors)
1064 RTPrintf("tstVD: SUCCESS\n");
1065 else
1066 RTPrintf("tstVD: FAILURE - %d errors\n", g_cErrors);
1067
1068 return !!g_cErrors;
1069}
1070
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette