VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/testcase/vditool.cpp@ 2418

Last change on this file since 2418 was 729, checked in by vboxsync, 18 years ago

Renamed to match target names.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 11.3 KB
Line 
1/** @file
2 *
3 * VBox HDD container maintenance/conversion utility
4 */
5
6/*
7 * Copyright (C) 2006 InnoTek Systemberatung GmbH
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 as published by the Free Software Foundation,
13 * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE
14 * distribution. VirtualBox OSE is distributed in the hope that it will
15 * be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * If you received this file as part of a commercial VirtualBox
18 * distribution, then only the terms of your commercial VirtualBox
19 * license agreement apply instead of the previous paragraph.
20 */
21
22/*******************************************************************************
23* Header Files *
24*******************************************************************************/
25#include <VBox/VBoxHDD.h>
26#include <iprt/alloc.h>
27#include <iprt/file.h>
28#include <iprt/stream.h>
29#include <iprt/string.h>
30#include <iprt/runtime.h>
31#include <VBox/err.h>
32
33#include <stdlib.h>
34
35
36
37static void ascii2upper(char *psz)
38{
39 for (;*psz; psz++)
40 if (*psz >= 'a' && *psz <= 'z')
41 *psz += 'A' - 'z';
42}
43
44static int UsageExit()
45{
46 RTPrintf("Usage: vditool <Command> [Params]\n" \
47 "Commands and params:\n" \
48 " NEW Filename Mbytes - create new image;\n" \
49 " DD Filename DDFilename - create new image from DD format image;\n" \
50 " CONVERT Filename - convert VDI image from old format;\n" \
51 " DUMP Filename - debug dump;\n" \
52 " RESETGEO Filename - reset geometry information;\n" \
53 " COPY FromImage ToImage - make image copy;\n" \
54 " COPYDD FromImage DDFilename - make a DD copy of the image;\n" \
55 " SHRINK Filename - optimize (reduce) VDI image size.\n");
56 return 1;
57}
58
59static int SyntaxError(const char *pszMsg)
60{
61 RTPrintf("Syntax error: %s\n\n", pszMsg);
62 UsageExit();
63 return 1;
64}
65
66/**
67 * Prints a done message indicating success or failure.
68 * @returns rc
69 * @param rc Status code.
70 */
71static int PrintDone(int rc)
72{
73 if (rc == VINF_SUCCESS)
74 RTPrintf("The operation completed successfully!\n");
75 else if (VBOX_SUCCESS(rc))
76 RTPrintf("The operation completed successfully! (rc=%Rrc)\n", rc);
77 else
78 RTPrintf("FAILURE: %Rrf (%Rrc)\n", rc, rc);
79 return rc;
80}
81
82static int NewImage(const char *pszFilename, uint32_t cMBs)
83{
84 RTPrintf("Creating VDI: file=\"%s\" size=%u MB...\n",
85 pszFilename, cMBs);
86 int rc = VDICreateBaseImage(pszFilename,
87 VDI_IMAGE_TYPE_NORMAL,
88 (uint64_t)cMBs * (uint64_t)(1024 * 1024),
89 "Newly created test image", NULL, NULL);
90 return PrintDone(rc);
91}
92
93static int ConvertDDImage(const char *pszFilename, const char *pszDDFilename)
94{
95 RTPrintf("Converting VDI: from DD image file=\"%s\" to file=\"%s\"...\n",
96 pszDDFilename, pszFilename);
97
98 /* open raw image file. */
99 RTFILE File;
100 int rc = RTFileOpen(&File, pszDDFilename, RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE);
101 if (VBOX_FAILURE(rc))
102 {
103 RTPrintf("File=\"%s\" open error: %Rrf\n", pszDDFilename, rc);
104 return rc;
105 }
106
107 /* get image size. */
108 uint64_t cbFile;
109 rc = RTFileGetSize(File, &cbFile);
110 if (VBOX_SUCCESS(rc))
111 {
112 RTPrintf("Creating fixed image with size %u Bytes...\n", (unsigned)cbFile);
113 rc = VDICreateBaseImage(pszFilename,
114 VDI_IMAGE_TYPE_FIXED,
115 cbFile,
116 "Converted from DD test image", NULL, NULL);
117 PrintDone(rc);
118 if (VBOX_SUCCESS(rc))
119 {
120 RTPrintf("Writing data...\n");
121 PVDIDISK pVdi = VDIDiskCreate();
122 rc = VDIDiskOpenImage(pVdi, pszFilename, VDI_OPEN_FLAGS_NORMAL);
123 if (VBOX_SUCCESS(rc))
124 {
125 /* alloc work buffer. */
126 void *pvBuf = RTMemAlloc(VDIDiskGetBufferSize(pVdi));
127 if (pvBuf)
128 {
129 uint64_t off = 0;
130 while (off < cbFile)
131 {
132 unsigned cbRead = 0;
133 rc = RTFileRead(File, pvBuf, VDIDiskGetBufferSize(pVdi), &cbRead);
134 if (VBOX_FAILURE(rc) || !cbRead)
135 break;
136 rc = VDIDiskWrite(pVdi, off, pvBuf, cbRead);
137 if (VBOX_FAILURE(rc))
138 break;
139 off += cbRead;
140 }
141
142 RTMemFree(pvBuf);
143 }
144 else
145 rc = VERR_NO_MEMORY;
146
147 VDIDiskCloseImage(pVdi);
148 }
149
150 if (VBOX_FAILURE(rc))
151 {
152 /* delete image on error */
153 VDIDeleteImage(pszFilename);
154 }
155 PrintDone(rc);
156 }
157 }
158 RTFileClose(File);
159
160 return rc;
161}
162
163static DECLCALLBACK(int) ProcessCallback(PVM pVM, unsigned uPercent, void *pvUser)
164{
165 unsigned *pPercent = (unsigned *)pvUser;
166
167 if (*pPercent != uPercent)
168 {
169 *pPercent = uPercent;
170 RTPrintf(".");
171 if ((uPercent % 10) == 0 && uPercent)
172 RTPrintf("%d%%", uPercent);
173 RTStrmFlush(g_pStdOut);
174 }
175
176 return VINF_SUCCESS;
177}
178
179static int ConvertOldImage(const char *pszFilename)
180{
181 RTPrintf("Converting VDI image file=\"%s\" to a new format...\n"
182 "progress: 0%%",
183 pszFilename);
184 unsigned uPercent = 0;
185 int rc = VDIConvertImage(pszFilename, ProcessCallback, &uPercent);
186 RTPrintf("\n");
187 return PrintDone(rc);
188}
189
190static int DumpImage(const char *pszFilename)
191{
192 RTPrintf("Dumping VDI image file=\"%s\" into the log file...\n", pszFilename);
193 PVDIDISK pVdi = VDIDiskCreate();
194 int rc = VDIDiskOpenImage(pVdi, pszFilename, VDI_OPEN_FLAGS_READONLY);
195 if (VBOX_SUCCESS(rc))
196 {
197 VDIDiskDumpImages(pVdi);
198 VDIDiskCloseAllImages(pVdi);
199 }
200 return PrintDone(rc);
201}
202
203static int ResetImageGeometry(const char *pszFilename)
204{
205 RTPrintf("Resetting geometry info of VDI image file=\"%s\"\n", pszFilename);
206 PVDIDISK pVdi = VDIDiskCreate();
207 int rc = VDIDiskOpenImage(pVdi, pszFilename, VDI_OPEN_FLAGS_NORMAL);
208 if (VBOX_SUCCESS(rc))
209 {
210 rc = VDIDiskSetGeometry(pVdi, 0, 0, 0);
211 if (VBOX_SUCCESS(rc))
212 rc = VDIDiskSetTranslation(pVdi, PDMBIOSTRANSLATION_AUTO);
213 }
214 VDIDiskCloseImage(pVdi);
215 return PrintDone(rc);
216}
217
218static int CopyImage(const char *pszDstFile, const char *pszSrcFile)
219{
220 RTPrintf("Copying VDI image file=\"%s\" to image file=\"%s\"...\n"
221 "progress: 0%%",
222 pszSrcFile, pszDstFile);
223 unsigned uPrecent = 0;
224 int rc = VDICopyImage(pszDstFile, pszSrcFile, NULL, ProcessCallback, &uPrecent);
225 RTPrintf("\n");
226 return PrintDone(rc);
227}
228
229static int CopyToDD(const char *pszDstFile, const char *pszSrcFile)
230{
231 RTPrintf("Copying VDI image file=\"%s\" to DD file=\"%s\"...\n",
232 pszSrcFile, pszDstFile);
233 PVDIDISK pVdi = VDIDiskCreate();
234 int rc = VDIDiskOpenImage(pVdi, pszSrcFile, VDI_OPEN_FLAGS_NORMAL);
235 if (VBOX_SUCCESS(rc))
236 {
237 RTFILE FileDst;
238 rc = RTFileOpen(&FileDst, pszDstFile, RTFILE_O_CREATE | RTFILE_O_READWRITE | RTFILE_O_DENY_WRITE);
239 if (VBOX_SUCCESS(rc))
240 {
241 uint64_t cbSrc = VDIDiskGetSize(pVdi);
242 const unsigned cbBuf = VDIDiskGetBlockSize(pVdi); /* or perhaps VDIDiskGetBufferSize(pVdi)? */
243 void *pvBuf = RTMemAlloc(cbBuf);
244 if (pvBuf)
245 {
246 uint64_t off = 0;
247 while (off < cbSrc)
248 {
249 rc = VDIDiskRead(pVdi, off, pvBuf, cbBuf);
250 if (VBOX_FAILURE(rc))
251 break;
252 rc = RTFileWrite(FileDst, pvBuf, cbBuf, NULL);
253 if (VBOX_FAILURE(rc))
254 break;
255 off += cbBuf;
256 }
257 RTMemFree(pvBuf);
258 }
259 RTFileClose(FileDst);
260 }
261 }
262 VDIDiskCloseImage(pVdi);
263 return PrintDone(rc);
264}
265
266static int ShrinkImage(const char *pszFilename)
267{
268 RTPrintf("Shrinking VDI image file=\"%s\"...\n"
269 "progress: 0%%",
270 pszFilename);
271 unsigned uPrecent;
272 int rc = VDIShrinkImage(pszFilename, ProcessCallback, &uPrecent);
273 RTPrintf("\n");
274 return PrintDone(rc);
275}
276
277int main(int argc, char **argv)
278{
279 putenv("VBOX_LOG_DEST=stdout");
280 putenv("VBOX_LOG_FLAGS=");
281
282 RTR3Init();
283 RTPrintf("vditool Copyright (c) 2004-2005 InnoTek Systemberatung GmbH.\n\n");
284
285 /*
286 * Do cmd line parsing.
287 */
288 if (argc < 2)
289 return UsageExit();
290
291 char szCmd[16];
292 if (strlen(argv[1]) >= sizeof(szCmd))
293 return SyntaxError("Invalid command!");
294 strcpy(szCmd, argv[1]);
295 ascii2upper(szCmd);
296
297 PRTLOGGER pLogger;
298 static const char * const s_apszGroups[] = VBOX_LOGGROUP_NAMES;
299 int rc = RTLogCreate(&pLogger, 0, "all",
300 NULL, ELEMENTS(s_apszGroups), s_apszGroups,
301 RTLOGDEST_STDOUT, NULL);
302 RTLogRelSetDefaultInstance(pLogger);
303
304 if (strcmp(szCmd, "NEW") == 0)
305 {
306 if (argc != 4)
307 return SyntaxError("Invalid argument count!");
308
309 uint32_t cMBs;
310 rc = RTStrToUInt32Ex(argv[3], NULL, 10, &cMBs);
311 if (VBOX_FAILURE(rc))
312 return SyntaxError("Invalid number!");
313 if ( cMBs < 2
314 || cMBs > 1024*1024)
315 {
316 RTPrintf("error: Disk size %RU32 (MB) is not within the range %u-%u!\n",
317 cMBs, 2, 1024*1024);
318 return 1;
319 }
320
321 rc = NewImage(argv[2], cMBs);
322 }
323 else if (strcmp(szCmd, "DD") == 0)
324 {
325 if (argc != 4)
326 return SyntaxError("Invalid argument count!");
327 rc = ConvertDDImage(argv[2], argv[3]);
328 }
329 else if (strcmp(szCmd, "CONVERT") == 0)
330 {
331 if (argc != 3)
332 return SyntaxError("Invalid argument count!");
333 rc = ConvertOldImage(argv[2]);
334 }
335 else if (strcmp(szCmd, "DUMP") == 0)
336 {
337 if (argc != 3)
338 return SyntaxError("Invalid argument count!");
339 rc = DumpImage(argv[2]);
340 }
341 else if (strcmp(szCmd, "RESETGEO") == 0)
342 {
343 if (argc != 3)
344 return SyntaxError("Invalid argument count!");
345 rc = ResetImageGeometry(argv[2]);
346 }
347 else if (strcmp(szCmd, "COPY") == 0)
348 {
349 if (argc != 4)
350 return SyntaxError("Invalid argument count!");
351 rc = CopyImage(argv[3], argv[2]);
352 }
353 else if (strcmp(szCmd, "COPYDD") == 0)
354 {
355 if (argc != 4)
356 return SyntaxError("Invalid argument count!");
357 rc = CopyToDD(argv[3], argv[2]);
358 }
359 else if (strcmp(szCmd, "SHRINK") == 0)
360 {
361 if (argc != 3)
362 return SyntaxError("Invalid argument count!");
363 rc = ShrinkImage(argv[2]);
364 }
365 else
366 return SyntaxError("Invalid command!");
367
368 RTLogFlush(NULL);
369 return !VBOX_SUCCESS(rc);
370}
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