VirtualBox

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

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

fixed OSE headers

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 13.6 KB
Line 
1/** @file
2 *
3 * VBox HDD container maintenance/conversion 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/*******************************************************************************
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/initterm.h>
31#include <VBox/err.h>
32#include <VBox/log.h>
33
34#include <stdlib.h>
35
36
37
38static void ascii2upper(char *psz)
39{
40 for (;*psz; psz++)
41 if (*psz >= 'a' && *psz <= 'z')
42 *psz += 'A' - 'a';
43}
44
45static int UsageExit()
46{
47 RTPrintf("Usage: vditool <Command> [Params]\n"
48 "Commands and params:\n"
49 " NEW Filename Mbytes - create new image\n"
50#if 0
51 " DD Filename DDFilename - create new image from DD format image\n"
52 " CONVERT Filename - convert VDI image from old format\n"
53 " DUMP Filename - debug dump\n"
54 " RESETGEO Filename - reset geometry information\n"
55 " COPY FromImage ToImage - make image copy\n"
56 " COPYDD FromImage DDFilename - make a DD copy of the image\n"
57 " SHRINK Filename - optimize (reduce) VDI image size\n"
58#endif
59 );
60 return 1;
61}
62
63static int SyntaxError(const char *pszMsg)
64{
65 RTPrintf("Syntax error: %s\n\n", pszMsg);
66 UsageExit();
67 return 1;
68}
69
70/**
71 * Our internal functions use UTF8
72 */
73static int FilenameToUtf8(char **pszUtf8Filename, const char *pszFilename)
74{
75 int rc = RTStrCurrentCPToUtf8(pszUtf8Filename, pszFilename);
76 if (RT_FAILURE(rc))
77 RTPrintf("Error converting filename '%s' to UTF8! (rc=%Rrc)\n",
78 pszFilename, rc);
79 return rc;
80}
81
82/**
83 * Prints a done message indicating success or failure.
84 * @returns rc
85 * @param rc Status code.
86 */
87static int PrintDone(int rc)
88{
89 if (rc == VINF_SUCCESS)
90 RTPrintf("The operation completed successfully!\n");
91 else if (RT_SUCCESS(rc))
92 RTPrintf("The operation completed successfully! (rc=%Rrc)\n", rc);
93 else
94 RTPrintf("FAILURE: %Rrf (%Rrc)\n", rc, rc);
95 return rc;
96}
97
98static int NewImage(const char *pszFilename, uint32_t cMBs)
99{
100 RTPrintf("Creating VDI: file=\"%s\" size=%u MB...\n",
101 pszFilename, cMBs);
102
103 /* translate argv[] to UTF8 */
104 char *pszUtf8Filename;
105 int rc = FilenameToUtf8(&pszUtf8Filename, pszFilename);
106 if (RT_FAILURE(rc))
107 return rc;
108
109 PVBOXHDD hdd;
110 rc = VDCreate(NULL, &hdd);
111 if (RT_FAILURE(rc))
112 return PrintDone(rc);
113
114 PDMMEDIAGEOMETRY geo = { 0 }; /* auto-detect */
115 rc = VDCreateBase(hdd, "vdi", pszUtf8Filename,
116 (uint64_t)cMBs * _1M,
117 VD_IMAGE_FLAGS_NONE,
118 "Newly created test image",
119 &geo, &geo, NULL,
120 VD_OPEN_FLAGS_NORMAL,
121 NULL, NULL);
122 return PrintDone(rc);
123}
124
125#if 0
126static int ConvertDDImage(const char *pszFilename, const char *pszDDFilename)
127{
128 RTPrintf("Converting VDI: from DD image file=\"%s\" to file=\"%s\"...\n",
129 pszDDFilename, pszFilename);
130
131 /* translate argv[] to UTF8 */
132 char *pszUtf8Filename, *pszUtf8DDFilename;
133 int rc = FilenameToUtf8(&pszUtf8Filename, pszFilename);
134 if (RT_FAILURE(rc))
135 return rc;
136 rc = FilenameToUtf8(&pszUtf8DDFilename, pszDDFilename);
137 if (RT_FAILURE(rc))
138 return rc;
139
140 /* open raw image file. */
141 RTFILE File;
142 rc = RTFileOpen(&File, pszUtf8DDFilename, RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE);
143 if (RT_FAILURE(rc))
144 {
145 RTPrintf("File=\"%s\" open error: %Rrf\n", pszDDFilename, rc);
146 return rc;
147 }
148
149 /* get image size. */
150 uint64_t cbFile;
151 rc = RTFileGetSize(File, &cbFile);
152 if (RT_SUCCESS(rc))
153 {
154 RTPrintf("Creating fixed image with size %u Bytes...\n", (unsigned)cbFile);
155 rc = VDICreateBaseImage(pszUtf8Filename,
156 VDI_IMAGE_TYPE_FIXED,
157 cbFile,
158 "Converted from DD test image", NULL, NULL);
159 PrintDone(rc);
160 if (RT_SUCCESS(rc))
161 {
162 RTPrintf("Writing data...\n");
163 PVDIDISK pVdi = VDIDiskCreate();
164 rc = VDIDiskOpenImage(pVdi, pszUtf8Filename, VDI_OPEN_FLAGS_NORMAL);
165 if (RT_SUCCESS(rc))
166 {
167 /* alloc work buffer. */
168 void *pvBuf = RTMemAlloc(VDIDiskGetBufferSize(pVdi));
169 if (pvBuf)
170 {
171 uint64_t off = 0;
172 while (off < cbFile)
173 {
174 size_t cbRead = 0;
175 rc = RTFileRead(File, pvBuf, VDIDiskGetBufferSize(pVdi), &cbRead);
176 if (RT_FAILURE(rc) || !cbRead)
177 break;
178 rc = VDIDiskWrite(pVdi, off, pvBuf, cbRead);
179 if (RT_FAILURE(rc))
180 break;
181 off += cbRead;
182 }
183
184 RTMemFree(pvBuf);
185 }
186 else
187 rc = VERR_NO_MEMORY;
188
189 VDIDiskCloseImage(pVdi);
190 }
191
192 if (RT_FAILURE(rc))
193 {
194 /* delete image on error */
195 VDIDeleteImage(pszUtf8Filename);
196 }
197 PrintDone(rc);
198 }
199 }
200 RTFileClose(File);
201
202 return rc;
203}
204#endif
205
206#if 0
207static DECLCALLBACK(int) ProcessCallback(PVM pVM, unsigned uPercent, void *pvUser)
208{
209 unsigned *pPercent = (unsigned *)pvUser;
210
211 if (*pPercent != uPercent)
212 {
213 *pPercent = uPercent;
214 RTPrintf(".");
215 if ((uPercent % 10) == 0 && uPercent)
216 RTPrintf("%d%%", uPercent);
217 RTStrmFlush(g_pStdOut);
218 }
219
220 return VINF_SUCCESS;
221}
222#endif
223
224#if 0
225static int ConvertOldImage(const char *pszFilename)
226{
227 RTPrintf("Converting VDI image file=\"%s\" to a new format...\n"
228 "progress: 0%%",
229 pszFilename);
230
231 /* translate argv[] to UTF8 */
232 char *pszUtf8Filename;
233 int rc = FilenameToUtf8(&pszUtf8Filename, pszFilename);
234 if (RT_FAILURE(rc))
235 return rc;
236
237 unsigned uPercent = 0;
238 rc = VDIConvertImage(pszUtf8Filename, ProcessCallback, &uPercent);
239 RTPrintf("\n");
240 return PrintDone(rc);
241}
242#endif
243
244#if 0
245static int DumpImage(const char *pszFilename)
246{
247 RTPrintf("Dumping VDI image file=\"%s\" into the log file...\n", pszFilename);
248 PVDIDISK pVdi = VDIDiskCreate();
249
250 /* translate argv[] to UTF8 */
251 char *pszUtf8Filename;
252 int rc = FilenameToUtf8(&pszUtf8Filename, pszFilename);
253 if (RT_FAILURE(rc))
254 return rc;
255 rc = VDIDiskOpenImage(pVdi, pszUtf8Filename, VDI_OPEN_FLAGS_READONLY);
256 if (RT_SUCCESS(rc))
257 {
258 VDIDiskDumpImages(pVdi);
259 VDIDiskCloseAllImages(pVdi);
260 }
261 return PrintDone(rc);
262}
263#endif
264
265#if 0
266static int ResetImageGeometry(const char *pszFilename)
267{
268 RTPrintf("Resetting geometry info of VDI image file=\"%s\"\n", pszFilename);
269 PVDIDISK pVdi = VDIDiskCreate();
270
271 /* translate argv[] to UTF8 */
272 char *pszUtf8Filename;
273 int rc = FilenameToUtf8(&pszUtf8Filename, pszFilename);
274 if (RT_FAILURE(rc))
275 return rc;
276
277 rc = VDIDiskOpenImage(pVdi, pszUtf8Filename, VDI_OPEN_FLAGS_NORMAL);
278 if (RT_SUCCESS(rc))
279 {
280 PDMMEDIAGEOMETRY LCHSGeometry = {0, 0, 0};
281 rc = VDIDiskSetLCHSGeometry(pVdi, &LCHSGeometry);
282 }
283 VDIDiskCloseImage(pVdi);
284 return PrintDone(rc);
285}
286#endif
287
288#if 0
289static int CopyImage(const char *pszDstFile, const char *pszSrcFile)
290{
291 RTPrintf("Copying VDI image file=\"%s\" to image file=\"%s\"...\n"
292 "progress: 0%%",
293 pszSrcFile, pszDstFile);
294
295 /* translate argv[] to UTF8 */
296 char *pszUtf8SrcFile, *pszUtf8DstFile;
297 int rc = FilenameToUtf8(&pszUtf8SrcFile, pszSrcFile);
298 if (RT_FAILURE(rc))
299 return rc;
300 rc = FilenameToUtf8(&pszUtf8DstFile, pszDstFile);
301 if (RT_FAILURE(rc))
302 return rc;
303
304 unsigned uPrecent = 0;
305 rc = VDICopyImage(pszUtf8DstFile, pszUtf8SrcFile, NULL, ProcessCallback, &uPrecent);
306 RTPrintf("\n");
307 return PrintDone(rc);
308}
309#endif
310
311#if 0
312static int CopyToDD(const char *pszDstFile, const char *pszSrcFile)
313{
314 RTPrintf("Copying VDI image file=\"%s\" to DD file=\"%s\"...\n",
315 pszSrcFile, pszDstFile);
316 PVDIDISK pVdi = VDIDiskCreate();
317
318 /* translate argv[] to UTF8 */
319 char *pszUtf8SrcFile, *pszUtf8DstFile;
320 int rc = FilenameToUtf8(&pszUtf8SrcFile, pszSrcFile);
321 if (RT_FAILURE(rc))
322 return rc;
323 rc = FilenameToUtf8(&pszUtf8DstFile, pszDstFile);
324 if (RT_FAILURE(rc))
325 return rc;
326
327 rc = VDIDiskOpenImage(pVdi, pszUtf8SrcFile, VDI_OPEN_FLAGS_NORMAL);
328 if (RT_SUCCESS(rc))
329 {
330 RTFILE FileDst;
331 rc = RTFileOpen(&FileDst, pszUtf8DstFile, RTFILE_O_CREATE | RTFILE_O_READWRITE | RTFILE_O_DENY_WRITE);
332 if (RT_SUCCESS(rc))
333 {
334 uint64_t cbSrc = VDIDiskGetSize(pVdi);
335 const unsigned cbBuf = VDIDiskGetBlockSize(pVdi); /* or perhaps VDIDiskGetBufferSize(pVdi)? */
336 void *pvBuf = RTMemAlloc(cbBuf);
337 if (pvBuf)
338 {
339 uint64_t off = 0;
340 while (off < cbSrc)
341 {
342 rc = VDIDiskRead(pVdi, off, pvBuf, cbBuf);
343 if (RT_FAILURE(rc))
344 break;
345 rc = RTFileWrite(FileDst, pvBuf, cbBuf, NULL);
346 if (RT_FAILURE(rc))
347 break;
348 off += cbBuf;
349 }
350 RTMemFree(pvBuf);
351 }
352 RTFileClose(FileDst);
353 }
354 }
355 VDIDiskCloseImage(pVdi);
356 return PrintDone(rc);
357}
358#endif
359
360#if 0
361static int ShrinkImage(const char *pszFilename)
362{
363 RTPrintf("Shrinking VDI image file=\"%s\"...\n"
364 "progress: 0%%",
365 pszFilename);
366
367 /* translate argv[] to UTF8 */
368 char *pszUtf8Filename;
369 int rc = FilenameToUtf8(&pszUtf8Filename, pszFilename);
370 if (RT_FAILURE(rc))
371 return rc;
372
373 unsigned uPrecent;
374 rc = VDIShrinkImage(pszUtf8Filename, ProcessCallback, &uPrecent);
375 RTPrintf("\n");
376 return PrintDone(rc);
377}
378#endif
379
380int main(int argc, char **argv)
381{
382 putenv((char*)"VBOX_LOG_DEST=stdout");
383 putenv((char*)"VBOX_LOG_FLAGS=");
384
385 RTR3Init();
386 RTPrintf("vditool -- for internal use only!\n"
387 "Copyright (c) 2009 Sun Microsystems, Inc.\n\n");
388
389 /*
390 * Do cmd line parsing.
391 */
392 if (argc < 2)
393 return UsageExit();
394
395 char szCmd[16];
396 if (strlen(argv[1]) >= sizeof(szCmd))
397 return SyntaxError("Invalid command!");
398 strcpy(szCmd, argv[1]);
399 ascii2upper(szCmd);
400
401 PRTLOGGER pLogger;
402 static const char * const s_apszGroups[] = VBOX_LOGGROUP_NAMES;
403 int rc = RTLogCreate(&pLogger, 0, "all",
404 NULL, RT_ELEMENTS(s_apszGroups), s_apszGroups,
405 RTLOGDEST_STDOUT, NULL);
406 RTLogRelSetDefaultInstance(pLogger);
407
408 if (strcmp(szCmd, "NEW") == 0)
409 {
410 if (argc != 4)
411 return SyntaxError("Invalid argument count!");
412
413 uint32_t cMBs;
414 rc = RTStrToUInt32Ex(argv[3], NULL, 10, &cMBs);
415 if (RT_FAILURE(rc))
416 return SyntaxError("Invalid number!");
417 if (cMBs < 2 || cMBs > _1M)
418 {
419 RTPrintf("error: Disk size %RU32 (MB) is not within the range %u-%u!\n",
420 cMBs, 2, _1M);
421 return 1;
422 }
423
424 rc = NewImage(argv[2], cMBs);
425 }
426#if 0
427 else if (strcmp(szCmd, "DD") == 0)
428 {
429 if (argc != 4)
430 return SyntaxError("Invalid argument count!");
431 rc = ConvertDDImage(argv[2], argv[3]);
432 }
433 else if (strcmp(szCmd, "CONVERT") == 0)
434 {
435 if (argc != 3)
436 return SyntaxError("Invalid argument count!");
437 rc = ConvertOldImage(argv[2]);
438 }
439 else if (strcmp(szCmd, "DUMP") == 0)
440 {
441 if (argc != 3)
442 return SyntaxError("Invalid argument count!");
443 rc = DumpImage(argv[2]);
444 }
445 else if (strcmp(szCmd, "RESETGEO") == 0)
446 {
447 if (argc != 3)
448 return SyntaxError("Invalid argument count!");
449 rc = ResetImageGeometry(argv[2]);
450 }
451 else if (strcmp(szCmd, "COPY") == 0)
452 {
453 if (argc != 4)
454 return SyntaxError("Invalid argument count!");
455 rc = CopyImage(argv[3], argv[2]);
456 }
457 else if (strcmp(szCmd, "COPYDD") == 0)
458 {
459 if (argc != 4)
460 return SyntaxError("Invalid argument count!");
461 rc = CopyToDD(argv[3], argv[2]);
462 }
463 else if (strcmp(szCmd, "SHRINK") == 0)
464 {
465 if (argc != 3)
466 return SyntaxError("Invalid argument count!");
467 rc = ShrinkImage(argv[2]);
468 }
469#endif
470 else
471 return SyntaxError("Invalid command!");
472
473 RTLogFlush(NULL);
474 return !RT_SUCCESS(rc);
475}
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