VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/testcase/tstVDCopy.cpp@ 27137

Last change on this file since 27137 was 22982, checked in by vboxsync, 15 years ago

VBoxHDD: Add new parameter to VDGetFormat which takes a list of interfaces. Backends need the async I/O interface if VBOX_WITH_NEW_IO_CODE is enabled to access the file

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 5.2 KB
Line 
1/** @file
2 *
3 * Simple VBox HDD container test utility. Compares two images and prints
4 * differences. Mainly used to check cloning of disk images.
5 */
6
7/*
8 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#include <VBox/err.h>
24#include <VBox/VBoxHDD.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/* from VBoxHDD.cpp */
35#define VD_MERGE_BUFFER_SIZE (16 * _1M)
36
37/*******************************************************************************
38* Global Variables *
39*******************************************************************************/
40/** The error count. */
41unsigned g_cErrors = 0;
42
43
44static void tstVDError(void *pvUser, int rc, RT_SRC_POS_DECL,
45 const char *pszFormat, va_list va)
46{
47 g_cErrors++;
48 RTPrintf("tstVD: Error %Rrc at %s:%u (%s): ", rc, RT_SRC_POS_ARGS);
49 RTPrintfV(pszFormat, va);
50 RTPrintf("\n");
51}
52
53int main(int argc, char *argv[])
54{
55 int rc;
56
57 RTR3Init();
58
59 if (argc != 3)
60 {
61 RTPrintf("Usage: ./tstVDCopy <hdd1> <hdd2>\n");
62 return 1;
63 }
64
65 RTPrintf("tstVDCopy: TESTING...\n");
66
67 PVBOXHDD pVD1 = NULL;
68 PVBOXHDD pVD2 = NULL;
69 PVDINTERFACE pVDIfs = NULL;
70 VDINTERFACE VDIError;
71 VDINTERFACEERROR VDIErrorCallbacks;
72 char *pszVD1 = NULL;
73 char *pszVD2 = NULL;
74 char *pbBuf1 = NULL;
75 char *pbBuf2 = NULL;
76
77#define CHECK(str) \
78 do \
79 { \
80 if (RT_FAILURE(rc)) \
81 { \
82 RTPrintf("%s rc=%Rrc\n", str, rc); \
83 if (pVD1) \
84 VDCloseAll(pVD1); \
85 if (pVD2) \
86 VDCloseAll(pVD2); \
87 return rc; \
88 } \
89 } while (0)
90
91 pbBuf1 = (char *)RTMemAllocZ(VD_MERGE_BUFFER_SIZE);
92 pbBuf2 = (char *)RTMemAllocZ(VD_MERGE_BUFFER_SIZE);
93
94 /* Create error interface. */
95 VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
96 VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
97 VDIErrorCallbacks.pfnError = tstVDError;
98
99 rc = VDInterfaceAdd(&VDIError, "tstVD_Error", VDINTERFACETYPE_ERROR, &VDIErrorCallbacks,
100 NULL, &pVDIfs);
101 AssertRC(rc);
102
103 rc = VDGetFormat(NULL, argv[1], &pszVD1);
104 CHECK("VDGetFormat() hdd1");
105
106 rc = VDGetFormat(NULL, argv[2], &pszVD2);
107 CHECK("VDGetFormat() hdd2");
108
109 rc = VDCreate(&VDIError, &pVD1);
110 CHECK("VDCreate() hdd1");
111
112 rc = VDCreate(&VDIError, &pVD2);
113 CHECK("VDCreate() hdd1");
114
115 rc = VDOpen(pVD1, pszVD1, argv[1], VD_OPEN_FLAGS_NORMAL, NULL);
116 CHECK("VDOpen() hdd1");
117
118 rc = VDOpen(pVD2, pszVD2, argv[2], VD_OPEN_FLAGS_NORMAL, NULL);
119 CHECK("VDOpen() hdd2");
120
121 uint64_t cbSize1 = 0;
122 uint64_t cbSize2 = 0;
123
124 cbSize1 = VDGetSize(pVD1, 0);
125 Assert(cbSize1 != 0);
126 cbSize2 = VDGetSize(pVD1, 0);
127 Assert(cbSize1 != 0);
128
129 if (cbSize1 == cbSize2)
130 {
131 uint64_t uOffCurr = 0;
132
133 /* Compare block by block. */
134 while (uOffCurr < cbSize1)
135 {
136 size_t cbRead = RT_MIN((cbSize1 - uOffCurr), VD_MERGE_BUFFER_SIZE);
137
138 rc = VDRead(pVD1, uOffCurr, pbBuf1, cbRead);
139 CHECK("VDRead() hdd1");
140
141 rc = VDRead(pVD2, uOffCurr, pbBuf2, cbRead);
142 CHECK("VDRead() hdd2");
143
144 if (memcmp(pbBuf1, pbBuf2, cbRead))
145 {
146 RTPrintf("tstVDCopy: Images differ uOffCurr=%llu\n", uOffCurr);
147 /* Do byte by byte comaprison. */
148 for (size_t i = 0; i < cbRead; i++)
149 {
150 if (pbBuf1[i] != pbBuf2[i])
151 {
152 RTPrintf("tstVDCopy: First different byte is at offset %llu\n", uOffCurr + i);
153 break;
154 }
155 }
156 break;
157 }
158
159 uOffCurr += cbRead;
160 }
161 }
162 else
163 RTPrintf("tstVDCopy: Images have different size hdd1=%llu hdd2=%llu\n", cbSize1, cbSize2);
164
165 VDClose(pVD1, false);
166 CHECK("VDClose() hdd1");
167
168 VDClose(pVD2, false);
169 CHECK("VDClose() hdd2");
170
171 VDDestroy(pVD1);
172 VDDestroy(pVD2);
173 RTMemFree(pbBuf1);
174 RTMemFree(pbBuf2);
175#undef CHECK
176
177 rc = VDShutdown();
178 if (RT_FAILURE(rc))
179 {
180 RTPrintf("tstVD-2: unloading backends failed! rc=%Rrc\n", rc);
181 g_cErrors++;
182 }
183 /*
184 * Summary
185 */
186 if (!g_cErrors)
187 RTPrintf("tstVD-2: SUCCESS\n");
188 else
189 RTPrintf("tstVD-2: FAILURE - %d errors\n", g_cErrors);
190
191 return !!g_cErrors;
192}
193
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