VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxManage/VBoxManageDisk.cpp@ 23454

Last change on this file since 23454 was 23223, checked in by vboxsync, 15 years ago

API: big medium handling change and lots of assorted other cleanups and fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 57.2 KB
Line 
1/* $Id: VBoxManageDisk.cpp 23223 2009-09-22 15:50:03Z vboxsync $ */
2/** @file
3 * VBoxManage - The disk delated commands.
4 */
5
6/*
7 * Copyright (C) 2006-2009 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#ifndef VBOX_ONLY_DOCS
23
24/*******************************************************************************
25* Header Files *
26*******************************************************************************/
27#include <VBox/com/com.h>
28#include <VBox/com/array.h>
29#include <VBox/com/ErrorInfo.h>
30#include <VBox/com/errorprint.h>
31#include <VBox/com/VirtualBox.h>
32
33#include <iprt/asm.h>
34#include <iprt/file.h>
35#include <iprt/path.h>
36#include <iprt/param.h>
37#include <iprt/stream.h>
38#include <iprt/string.h>
39#include <iprt/ctype.h>
40#include <iprt/getopt.h>
41#include <VBox/log.h>
42#include <VBox/VBoxHDD.h>
43
44#include "VBoxManage.h"
45using namespace com;
46
47
48// funcs
49///////////////////////////////////////////////////////////////////////////////
50
51
52static DECLCALLBACK(void) handleVDError(void *pvUser, int rc, RT_SRC_POS_DECL, const char *pszFormat, va_list va)
53{
54 RTPrintf("ERROR: ");
55 RTPrintfV(pszFormat, va);
56 RTPrintf("\n");
57 RTPrintf("Error code %Rrc at %s(%u) in function %s\n", rc, RT_SRC_POS_ARGS);
58}
59
60
61static int parseDiskVariant(const char *psz, MediumVariant_T *pDiskVariant)
62{
63 int rc = VINF_SUCCESS;
64 unsigned DiskVariant = (unsigned)(*pDiskVariant);
65 while (psz && *psz && RT_SUCCESS(rc))
66 {
67 size_t len;
68 const char *pszComma = strchr(psz, ',');
69 if (pszComma)
70 len = pszComma - psz;
71 else
72 len = strlen(psz);
73 if (len > 0)
74 {
75 // Parsing is intentionally inconsistent: "standard" resets the
76 // variant, whereas the other flags are cumulative.
77 if (!RTStrNICmp(psz, "standard", len))
78 DiskVariant = MediumVariant_Standard;
79 else if ( !RTStrNICmp(psz, "fixed", len)
80 || !RTStrNICmp(psz, "static", len))
81 DiskVariant |= MediumVariant_Fixed;
82 else if (!RTStrNICmp(psz, "Diff", len))
83 DiskVariant |= MediumVariant_Diff;
84 else if (!RTStrNICmp(psz, "split2g", len))
85 DiskVariant |= MediumVariant_VmdkSplit2G;
86 else if ( !RTStrNICmp(psz, "stream", len)
87 || !RTStrNICmp(psz, "streamoptimized", len))
88 DiskVariant |= MediumVariant_VmdkStreamOptimized;
89 else if (!RTStrNICmp(psz, "esx", len))
90 DiskVariant |= MediumVariant_VmdkESX;
91 else
92 rc = VERR_PARSE_ERROR;
93 }
94 if (pszComma)
95 psz += len + 1;
96 else
97 psz += len;
98 }
99
100 if (RT_SUCCESS(rc))
101 *pDiskVariant = (MediumVariant_T)DiskVariant;
102 return rc;
103}
104
105static int parseDiskType(const char *psz, MediumType_T *pDiskType)
106{
107 int rc = VINF_SUCCESS;
108 MediumType_T DiskType = MediumType_Normal;
109 if (!RTStrICmp(psz, "normal"))
110 DiskType = MediumType_Normal;
111 else if (!RTStrICmp(psz, "immutable"))
112 DiskType = MediumType_Immutable;
113 else if (!RTStrICmp(psz, "writethrough"))
114 DiskType = MediumType_Writethrough;
115 else
116 rc = VERR_PARSE_ERROR;
117
118 if (RT_SUCCESS(rc))
119 *pDiskType = DiskType;
120 return rc;
121}
122
123/** @todo move this into getopt, as getting bool values is generic */
124static int parseBool(const char *psz, bool *pb)
125{
126 int rc = VINF_SUCCESS;
127 if ( !RTStrICmp(psz, "on")
128 || !RTStrICmp(psz, "yes")
129 || !RTStrICmp(psz, "true")
130 || !RTStrICmp(psz, "1")
131 || !RTStrICmp(psz, "enable")
132 || !RTStrICmp(psz, "enabled"))
133 {
134 *pb = true;
135 }
136 else if ( !RTStrICmp(psz, "off")
137 || !RTStrICmp(psz, "no")
138 || !RTStrICmp(psz, "false")
139 || !RTStrICmp(psz, "0")
140 || !RTStrICmp(psz, "disable")
141 || !RTStrICmp(psz, "disabled"))
142 {
143 *pb = false;
144 }
145 else
146 rc = VERR_PARSE_ERROR;
147
148 return rc;
149}
150
151static const RTGETOPTDEF g_aCreateHardDiskOptions[] =
152{
153 { "--filename", 'f', RTGETOPT_REQ_STRING },
154 { "-filename", 'f', RTGETOPT_REQ_STRING }, // deprecated
155 { "--size", 's', RTGETOPT_REQ_UINT64 },
156 { "-size", 's', RTGETOPT_REQ_UINT64 }, // deprecated
157 { "--format", 'o', RTGETOPT_REQ_STRING },
158 { "-format", 'o', RTGETOPT_REQ_STRING }, // deprecated
159 { "--static", 'F', RTGETOPT_REQ_NOTHING },
160 { "-static", 'F', RTGETOPT_REQ_NOTHING }, // deprecated
161 { "--variant", 'm', RTGETOPT_REQ_STRING },
162 { "-variant", 'm', RTGETOPT_REQ_STRING }, // deprecated
163 { "--type", 't', RTGETOPT_REQ_STRING },
164 { "-type", 't', RTGETOPT_REQ_STRING }, // deprecated
165 { "--comment", 'c', RTGETOPT_REQ_STRING },
166 { "-comment", 'c', RTGETOPT_REQ_STRING }, // deprecated
167 { "--remember", 'r', RTGETOPT_REQ_NOTHING },
168 { "-remember", 'r', RTGETOPT_REQ_NOTHING }, // deprecated
169 { "--register", 'r', RTGETOPT_REQ_NOTHING }, // deprecated (inofficial)
170 { "-register", 'r', RTGETOPT_REQ_NOTHING }, // deprecated
171};
172
173int handleCreateHardDisk(HandlerArg *a)
174{
175 HRESULT rc;
176 int vrc;
177 Bstr filename;
178 uint64_t sizeMB = 0;
179 Bstr format = "VDI";
180 MediumVariant_T DiskVariant = MediumVariant_Standard;
181 Bstr comment;
182 bool fRemember = false;
183 MediumType_T DiskType = MediumType_Normal;
184
185 int c;
186 RTGETOPTUNION ValueUnion;
187 RTGETOPTSTATE GetState;
188 // start at 0 because main() has hacked both the argc and argv given to us
189 RTGetOptInit(&GetState, a->argc, a->argv, g_aCreateHardDiskOptions, RT_ELEMENTS(g_aCreateHardDiskOptions), 0, 0 /* fFlags */);
190 while ((c = RTGetOpt(&GetState, &ValueUnion)))
191 {
192 switch (c)
193 {
194 case 'f': // --filename
195 filename = ValueUnion.psz;
196 break;
197
198 case 's': // --size
199 sizeMB = ValueUnion.u64;
200 break;
201
202 case 'o': // --format
203 format = ValueUnion.psz;
204 break;
205
206 case 'F': // --static ("fixed"/"flat")
207 {
208 unsigned uDiskVariant = (unsigned)DiskVariant;
209 uDiskVariant |= MediumVariant_Fixed;
210 DiskVariant = (MediumVariant_T)uDiskVariant;
211 break;
212 }
213
214 case 'm': // --variant
215 vrc = parseDiskVariant(ValueUnion.psz, &DiskVariant);
216 if (RT_FAILURE(vrc))
217 return errorArgument("Invalid hard disk variant '%s'", ValueUnion.psz);
218 break;
219
220 case 'c': // --comment
221 comment = ValueUnion.psz;
222 break;
223
224 case 'r': // --remember
225 fRemember = true;
226 break;
227
228 case 't': // --type
229 vrc = parseDiskType(ValueUnion.psz, &DiskType);
230 if ( RT_FAILURE(vrc)
231 || (DiskType != MediumType_Normal && DiskType != MediumType_Writethrough))
232 return errorArgument("Invalid hard disk type '%s'", ValueUnion.psz);
233 break;
234
235 case VINF_GETOPT_NOT_OPTION:
236 return errorSyntax(USAGE_CREATEHD, "Invalid parameter '%s'", ValueUnion.psz);
237
238 default:
239 if (c > 0)
240 {
241 if (RT_C_IS_PRINT(c))
242 return errorSyntax(USAGE_CREATEHD, "Invalid option -%c", c);
243 else
244 return errorSyntax(USAGE_CREATEHD, "Invalid option case %i", c);
245 }
246 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
247 return errorSyntax(USAGE_CREATEHD, "unknown option: %s\n", ValueUnion.psz);
248 else if (ValueUnion.pDef)
249 return errorSyntax(USAGE_CREATEHD, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
250 else
251 return errorSyntax(USAGE_CREATEHD, "error: %Rrs", c);
252 }
253 }
254
255 /* check the outcome */
256 if (!filename || (sizeMB == 0))
257 return errorSyntax(USAGE_CREATEHD, "Parameters --filename and --size are required");
258
259 ComPtr<IMedium> hardDisk;
260 CHECK_ERROR(a->virtualBox, CreateHardDisk(format, filename, hardDisk.asOutParam()));
261 if (SUCCEEDED(rc) && hardDisk)
262 {
263 /* we will close the hard disk after the storage has been successfully
264 * created unless fRemember is set */
265 bool doClose = false;
266
267 if (!comment.isNull())
268 {
269 CHECK_ERROR(hardDisk,COMSETTER(Description)(comment));
270 }
271 ComPtr<IProgress> progress;
272 CHECK_ERROR(hardDisk, CreateBaseStorage(sizeMB, DiskVariant, progress.asOutParam()));
273 if (SUCCEEDED(rc) && progress)
274 {
275 showProgress(progress);
276 if (SUCCEEDED(rc))
277 {
278 LONG iRc;
279 progress->COMGETTER(ResultCode)(&iRc);
280 rc = iRc;
281 if (FAILED(rc))
282 {
283 com::ProgressErrorInfo info(progress);
284 if (info.isBasicAvailable())
285 RTPrintf("Error: failed to create hard disk. Error message: %lS\n", info.getText().raw());
286 else
287 RTPrintf("Error: failed to create hard disk. No error message available!\n");
288 }
289 else
290 {
291 doClose = !fRemember;
292
293 Bstr uuid;
294 CHECK_ERROR(hardDisk, COMGETTER(Id)(uuid.asOutParam()));
295
296 if (DiskType == MediumType_Writethrough)
297 {
298 CHECK_ERROR(hardDisk, COMSETTER(Type)(MediumType_Writethrough));
299 }
300
301 RTPrintf("Disk image created. UUID: %s\n", Utf8Str(uuid).raw());
302 }
303 }
304 }
305 if (doClose)
306 {
307 CHECK_ERROR(hardDisk, Close());
308 }
309 }
310 return SUCCEEDED(rc) ? 0 : 1;
311}
312
313#if 0 /* disabled until disk shrinking is implemented based on VBoxHDD */
314static DECLCALLBACK(int) hardDiskProgressCallback(PVM pVM, unsigned uPercent, void *pvUser)
315{
316 unsigned *pPercent = (unsigned *)pvUser;
317
318 if (*pPercent != uPercent)
319 {
320 *pPercent = uPercent;
321 RTPrintf(".");
322 if ((uPercent % 10) == 0 && uPercent)
323 RTPrintf("%d%%", uPercent);
324 RTStrmFlush(g_pStdOut);
325 }
326
327 return VINF_SUCCESS;
328}
329#endif
330
331static const RTGETOPTDEF g_aModifyHardDiskOptions[] =
332{
333 { "--type", 't', RTGETOPT_REQ_STRING },
334 { "-type", 't', RTGETOPT_REQ_STRING }, // deprecated
335 { "settype", 't', RTGETOPT_REQ_STRING }, // deprecated
336 { "--autoreset", 'z', RTGETOPT_REQ_STRING },
337 { "-autoreset", 'z', RTGETOPT_REQ_STRING }, // deprecated
338 { "autoreset", 'z', RTGETOPT_REQ_STRING }, // deprecated
339 { "--compact", 'c', RTGETOPT_REQ_NOTHING },
340 { "-compact", 'c', RTGETOPT_REQ_NOTHING }, // deprecated
341 { "compact", 'c', RTGETOPT_REQ_NOTHING }, // deprecated
342};
343
344int handleModifyHardDisk(HandlerArg *a)
345{
346 HRESULT rc;
347 int vrc;
348 ComPtr<IMedium> hardDisk;
349 MediumType_T DiskType;
350 bool AutoReset = false;
351 bool fModifyDiskType = false, fModifyAutoReset = false, fModifyCompact = false;;
352 const char *FilenameOrUuid = NULL;
353
354 int c;
355 RTGETOPTUNION ValueUnion;
356 RTGETOPTSTATE GetState;
357 // start at 0 because main() has hacked both the argc and argv given to us
358 RTGetOptInit(&GetState, a->argc, a->argv, g_aModifyHardDiskOptions, RT_ELEMENTS(g_aModifyHardDiskOptions), 0, 0 /* fFlags */);
359 while ((c = RTGetOpt(&GetState, &ValueUnion)))
360 {
361 switch (c)
362 {
363 case 't': // --type
364 vrc = parseDiskType(ValueUnion.psz, &DiskType);
365 if (RT_FAILURE(vrc))
366 return errorArgument("Invalid hard disk type '%s'", ValueUnion.psz);
367 fModifyDiskType = true;
368 break;
369
370 case 'z': // --autoreset
371 vrc = parseBool(ValueUnion.psz, &AutoReset);
372 if (RT_FAILURE(vrc))
373 return errorArgument("Invalid autoreset parameter '%s'", ValueUnion.psz);
374 fModifyAutoReset = true;
375 break;
376
377 case 'c': // --compact
378 fModifyCompact = true;
379 break;
380
381 case VINF_GETOPT_NOT_OPTION:
382 if (!FilenameOrUuid)
383 FilenameOrUuid = ValueUnion.psz;
384 else
385 return errorSyntax(USAGE_CREATEHD, "Invalid parameter '%s'", ValueUnion.psz);
386 break;
387
388 default:
389 if (c > 0)
390 {
391 if (RT_C_IS_PRINT(c))
392 return errorSyntax(USAGE_MODIFYHD, "Invalid option -%c", c);
393 else
394 return errorSyntax(USAGE_MODIFYHD, "Invalid option case %i", c);
395 }
396 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
397 return errorSyntax(USAGE_MODIFYHD, "unknown option: %s\n", ValueUnion.psz);
398 else if (ValueUnion.pDef)
399 return errorSyntax(USAGE_MODIFYHD, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
400 else
401 return errorSyntax(USAGE_MODIFYHD, "error: %Rrs", c);
402 }
403 }
404
405 if (!FilenameOrUuid)
406 return errorSyntax(USAGE_MODIFYHD, "Disk name or UUID required");
407
408 if (!fModifyDiskType && !fModifyAutoReset && !fModifyCompact)
409 return errorSyntax(USAGE_MODIFYHD, "No operation specified");
410
411 /* first guess is that it's a UUID */
412 Guid uuid(FilenameOrUuid);
413 rc = a->virtualBox->GetHardDisk(uuid.toUtf16(), hardDisk.asOutParam());
414 /* no? then it must be a filename */
415 if (!hardDisk)
416 {
417 CHECK_ERROR(a->virtualBox, FindHardDisk(Bstr(FilenameOrUuid), hardDisk.asOutParam()));
418 if (FAILED(rc))
419 return 1;
420 }
421
422 if (fModifyDiskType)
423 {
424 /* hard disk must be registered */
425 if (SUCCEEDED(rc) && hardDisk)
426 {
427 MediumType_T hddType;
428 CHECK_ERROR(hardDisk, COMGETTER(Type)(&hddType));
429
430 if (hddType != DiskType)
431 CHECK_ERROR(hardDisk, COMSETTER(Type)(DiskType));
432 }
433 else
434 return errorArgument("Hard disk image not registered");
435 }
436
437 if (fModifyAutoReset)
438 {
439 CHECK_ERROR(hardDisk, COMSETTER(AutoReset)(AutoReset));
440 }
441
442 if (fModifyCompact)
443 {
444 bool unknown = false;
445 /* the hard disk image might not be registered */
446 if (!hardDisk)
447 {
448 unknown = true;
449 rc = a->virtualBox->OpenHardDisk(Bstr(FilenameOrUuid), AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), hardDisk.asOutParam());
450 if (rc == VBOX_E_FILE_ERROR)
451 {
452 char szFilenameAbs[RTPATH_MAX] = "";
453 int vrc = RTPathAbs(FilenameOrUuid, szFilenameAbs, sizeof(szFilenameAbs));
454 if (RT_FAILURE(vrc))
455 {
456 RTPrintf("Cannot convert filename \"%s\" to absolute path\n", FilenameOrUuid);
457 return 1;
458 }
459 CHECK_ERROR(a->virtualBox, OpenHardDisk(Bstr(szFilenameAbs), AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), hardDisk.asOutParam()));
460 }
461 else if (FAILED(rc))
462 CHECK_ERROR(a->virtualBox, OpenHardDisk(Bstr(FilenameOrUuid), AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), hardDisk.asOutParam()));
463 }
464 if (SUCCEEDED(rc) && hardDisk)
465 {
466 ComPtr<IProgress> progress;
467 CHECK_ERROR(hardDisk, Compact(progress.asOutParam()));
468 if (SUCCEEDED(rc))
469 {
470 showProgress(progress);
471 LONG iRc;
472 progress->COMGETTER(ResultCode)(&iRc);
473 rc = iRc;
474 }
475 if (FAILED(rc))
476 {
477 if (rc == E_NOTIMPL)
478 {
479 RTPrintf("Error: Compact hard disk operation is not implemented!\n");
480 RTPrintf("The functionality will be restored later.\n");
481 }
482 else if (rc == VBOX_E_NOT_SUPPORTED)
483 {
484 RTPrintf("Error: Compact hard disk operation for this format is not implemented yet!\n");
485 }
486 else
487 com::GluePrintRCMessage(rc);
488 }
489 if (unknown)
490 hardDisk->Close();
491 }
492 }
493
494 return SUCCEEDED(rc) ? 0 : 1;
495}
496
497static const RTGETOPTDEF g_aCloneHardDiskOptions[] =
498{
499 { "--format", 'o', RTGETOPT_REQ_STRING },
500 { "-format", 'o', RTGETOPT_REQ_STRING },
501 { "--static", 'F', RTGETOPT_REQ_NOTHING },
502 { "-static", 'F', RTGETOPT_REQ_NOTHING },
503 { "--existing", 'E', RTGETOPT_REQ_NOTHING },
504 { "--variant", 'm', RTGETOPT_REQ_STRING },
505 { "-variant", 'm', RTGETOPT_REQ_STRING },
506 { "--type", 't', RTGETOPT_REQ_STRING },
507 { "-type", 't', RTGETOPT_REQ_STRING },
508 { "--remember", 'r', RTGETOPT_REQ_NOTHING },
509 { "-remember", 'r', RTGETOPT_REQ_NOTHING },
510 { "--register", 'r', RTGETOPT_REQ_NOTHING },
511 { "-register", 'r', RTGETOPT_REQ_NOTHING },
512};
513
514int handleCloneHardDisk(HandlerArg *a)
515{
516 HRESULT rc;
517 int vrc;
518 Bstr src, dst;
519 Bstr format;
520 MediumVariant_T DiskVariant = MediumVariant_Standard;
521 bool fExisting = false;
522 bool fRemember = false;
523 bool fSetDiskType = false;
524 MediumType_T DiskType = MediumType_Normal;
525
526 int c;
527 RTGETOPTUNION ValueUnion;
528 RTGETOPTSTATE GetState;
529 // start at 0 because main() has hacked both the argc and argv given to us
530 RTGetOptInit(&GetState, a->argc, a->argv, g_aCloneHardDiskOptions, RT_ELEMENTS(g_aCloneHardDiskOptions), 0, 0 /* fFlags */);
531 while ((c = RTGetOpt(&GetState, &ValueUnion)))
532 {
533 switch (c)
534 {
535 case 'o': // --format
536 format = ValueUnion.psz;
537 break;
538
539 case 'F': // --static
540 {
541 unsigned uDiskVariant = (unsigned)DiskVariant;
542 uDiskVariant |= MediumVariant_Fixed;
543 DiskVariant = (MediumVariant_T)uDiskVariant;
544 break;
545 }
546
547 case 'E': // --existing
548 fExisting = true;
549 break;
550
551 case 'm': // --variant
552 vrc = parseDiskVariant(ValueUnion.psz, &DiskVariant);
553 if (RT_FAILURE(vrc))
554 return errorArgument("Invalid hard disk variant '%s'", ValueUnion.psz);
555 break;
556
557 case 'r': // --remember
558 fRemember = true;
559 break;
560
561 case 't': // --type
562 vrc = parseDiskType(ValueUnion.psz, &DiskType);
563 if (RT_FAILURE(vrc))
564 return errorArgument("Invalid hard disk type '%s'", ValueUnion.psz);
565 fSetDiskType = true;
566 break;
567
568 case VINF_GETOPT_NOT_OPTION:
569 if (src.isEmpty())
570 src = ValueUnion.psz;
571 else if (dst.isEmpty())
572 dst = ValueUnion.psz;
573 else
574 return errorSyntax(USAGE_CLONEHD, "Invalid parameter '%s'", ValueUnion.psz);
575 break;
576
577 default:
578 if (c > 0)
579 {
580 if (RT_C_IS_GRAPH(c))
581 return errorSyntax(USAGE_CLONEHD, "unhandled option: -%c", c);
582 else
583 return errorSyntax(USAGE_CLONEHD, "unhandled option: %i", c);
584 }
585 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
586 return errorSyntax(USAGE_CLONEHD, "unknown option: %s", ValueUnion.psz);
587 else if (ValueUnion.pDef)
588 return errorSyntax(USAGE_CLONEHD, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
589 else
590 return errorSyntax(USAGE_CLONEHD, "error: %Rrs", c);
591 }
592 }
593
594 if (src.isEmpty())
595 return errorSyntax(USAGE_CLONEHD, "Mandatory UUID or input file parameter missing");
596 if (dst.isEmpty())
597 return errorSyntax(USAGE_CLONEHD, "Mandatory output file parameter missing");
598 if (fExisting && (!format.isEmpty() || DiskVariant != MediumType_Normal))
599 return errorSyntax(USAGE_CLONEHD, "Specified options which cannot be used with --existing");
600
601 ComPtr<IMedium> srcDisk;
602 ComPtr<IMedium> dstDisk;
603 bool fSrcUnknown = false;
604 bool fDstUnknown = false;
605
606 /* first guess is that it's a UUID */
607 rc = a->virtualBox->GetHardDisk(src, srcDisk.asOutParam());
608 /* no? then it must be a filename */
609 if (FAILED (rc))
610 rc = a->virtualBox->FindHardDisk(src, srcDisk.asOutParam());
611 /* no? well, then it's an unknown image */
612 if (FAILED (rc))
613 {
614 rc = a->virtualBox->OpenHardDisk(src, AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), srcDisk.asOutParam());
615 if (rc == VBOX_E_FILE_ERROR)
616 {
617 char szFilenameAbs[RTPATH_MAX] = "";
618 int vrc = RTPathAbs(Utf8Str(src).c_str(), szFilenameAbs, sizeof(szFilenameAbs));
619 if (RT_FAILURE(vrc))
620 {
621 RTPrintf("Cannot convert filename \"%s\" to absolute path\n", Utf8Str(src).raw());
622 return 1;
623 }
624 CHECK_ERROR(a->virtualBox, OpenHardDisk(Bstr(szFilenameAbs), AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), srcDisk.asOutParam()));
625 }
626 else if (FAILED(rc))
627 CHECK_ERROR(a->virtualBox, OpenHardDisk(src, AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), srcDisk.asOutParam()));
628 if (SUCCEEDED (rc))
629 fSrcUnknown = true;
630 }
631
632 do
633 {
634 if (!SUCCEEDED(rc))
635 break;
636
637 /* open/create destination hard disk */
638 if (fExisting)
639 {
640 /* first guess is that it's a UUID */
641 rc = a->virtualBox->GetHardDisk(dst, dstDisk.asOutParam());
642 /* no? then it must be a filename */
643 if (FAILED (rc))
644 rc = a->virtualBox->FindHardDisk(dst, dstDisk.asOutParam());
645 /* no? well, then it's an unknown image */
646 if (FAILED (rc))
647 {
648 rc = a->virtualBox->OpenHardDisk(dst, AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), dstDisk.asOutParam());
649 if (rc == VBOX_E_FILE_ERROR)
650 {
651 char szFilenameAbs[RTPATH_MAX] = "";
652 int vrc = RTPathAbs(Utf8Str(dst).c_str(), szFilenameAbs, sizeof(szFilenameAbs));
653 if (RT_FAILURE(vrc))
654 {
655 RTPrintf("Cannot convert filename \"%s\" to absolute path\n", Utf8Str(dst).raw());
656 return 1;
657 }
658 CHECK_ERROR_BREAK(a->virtualBox, OpenHardDisk(Bstr(szFilenameAbs), AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), dstDisk.asOutParam()));
659 }
660 else if (FAILED(rc))
661 CHECK_ERROR_BREAK(a->virtualBox, OpenHardDisk(dst, AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), dstDisk.asOutParam()));
662 if (SUCCEEDED (rc))
663 fDstUnknown = true;
664 }
665 else
666 fRemember = true;
667 if (SUCCEEDED(rc))
668 {
669 /* Perform accessibility check now. */
670 MediumState_T state;
671 CHECK_ERROR_BREAK(dstDisk, COMGETTER(State)(&state));
672 }
673 CHECK_ERROR_BREAK(dstDisk, COMGETTER(Format) (format.asOutParam()));
674 }
675 else
676 {
677 /* use the format of the source hard disk if unspecified */
678 if (format.isEmpty())
679 CHECK_ERROR_BREAK(srcDisk, COMGETTER(Format) (format.asOutParam()));
680 CHECK_ERROR_BREAK(a->virtualBox, CreateHardDisk(format, dst, dstDisk.asOutParam()));
681 }
682
683 ComPtr<IProgress> progress;
684 CHECK_ERROR_BREAK(srcDisk, CloneTo(dstDisk, DiskVariant, NULL, progress.asOutParam()));
685
686 showProgress(progress);
687 LONG iRc;
688 progress->COMGETTER(ResultCode)(&iRc);
689 rc = iRc;
690 if (FAILED(rc))
691 {
692 com::ProgressErrorInfo info(progress);
693 if (info.isBasicAvailable())
694 RTPrintf("Error: failed to clone hard disk. Error message: %lS\n", info.getText().raw());
695 else
696 RTPrintf("Error: failed to clone hard disk. No error message available!\n");
697 break;
698 }
699
700 Bstr uuid;
701 CHECK_ERROR_BREAK(dstDisk, COMGETTER(Id)(uuid.asOutParam()));
702
703 RTPrintf("Clone hard disk created in format '%ls'. UUID: %s\n",
704 format.raw(), Utf8Str(uuid).raw());
705 }
706 while (0);
707
708 if (!fRemember && !dstDisk.isNull())
709 {
710 /* forget the created clone */
711 dstDisk->Close();
712 }
713 else if (fSetDiskType)
714 {
715 CHECK_ERROR(dstDisk, COMSETTER(Type)(DiskType));
716 }
717
718 if (fSrcUnknown)
719 {
720 /* close the unknown hard disk to forget it again */
721 srcDisk->Close();
722 }
723
724 return SUCCEEDED(rc) ? 0 : 1;
725}
726
727static const RTGETOPTDEF g_aConvertFromRawHardDiskOptions[] =
728{
729 { "--format", 'o', RTGETOPT_REQ_STRING },
730 { "-format", 'o', RTGETOPT_REQ_STRING },
731 { "--static", 'F', RTGETOPT_REQ_NOTHING },
732 { "-static", 'F', RTGETOPT_REQ_NOTHING },
733 { "--variant", 'm', RTGETOPT_REQ_STRING },
734 { "-variant", 'm', RTGETOPT_REQ_STRING },
735};
736
737int handleConvertFromRaw(int argc, char *argv[])
738{
739 int rc = VINF_SUCCESS;
740 bool fReadFromStdIn = false;
741 const char *format = "VDI";
742 const char *srcfilename = NULL;
743 const char *dstfilename = NULL;
744 const char *filesize = NULL;
745 unsigned uImageFlags = VD_IMAGE_FLAGS_NONE;
746 void *pvBuf = NULL;
747
748 int c;
749 RTGETOPTUNION ValueUnion;
750 RTGETOPTSTATE GetState;
751 // start at 0 because main() has hacked both the argc and argv given to us
752 RTGetOptInit(&GetState, argc, argv, g_aConvertFromRawHardDiskOptions, RT_ELEMENTS(g_aConvertFromRawHardDiskOptions), 0, 0 /* fFlags */);
753 while ((c = RTGetOpt(&GetState, &ValueUnion)))
754 {
755 switch (c)
756 {
757 case 'o': // --format
758 format = ValueUnion.psz;
759 break;
760
761 case 'm': // --variant
762 MediumVariant_T DiskVariant;
763 rc = parseDiskVariant(ValueUnion.psz, &DiskVariant);
764 if (RT_FAILURE(rc))
765 return errorArgument("Invalid hard disk variant '%s'", ValueUnion.psz);
766 /// @todo cleaner solution than assuming 1:1 mapping?
767 uImageFlags = (unsigned)DiskVariant;
768 break;
769
770 case VINF_GETOPT_NOT_OPTION:
771 if (!srcfilename)
772 {
773 srcfilename = ValueUnion.psz;
774#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_SOLARIS)
775 fReadFromStdIn = !strcmp(srcfilename, "stdin");
776#endif
777 }
778 else if (!dstfilename)
779 dstfilename = ValueUnion.psz;
780 else if (fReadFromStdIn && !filesize)
781 filesize = ValueUnion.psz;
782 else
783 return errorSyntax(USAGE_CONVERTFROMRAW, "Invalid parameter '%s'", ValueUnion.psz);
784 break;
785
786 default:
787 if (c > 0)
788 {
789 if (RT_C_IS_PRINT(c))
790 return errorSyntax(USAGE_CONVERTFROMRAW, "Invalid option -%c", c);
791 else
792 return errorSyntax(USAGE_CONVERTFROMRAW, "Invalid option case %i", c);
793 }
794 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
795 return errorSyntax(USAGE_CREATEHD, "unknown option: %s\n", ValueUnion.psz);
796 else if (ValueUnion.pDef)
797 return errorSyntax(USAGE_CONVERTFROMRAW, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
798 else
799 return errorSyntax(USAGE_CONVERTFROMRAW, "error: %Rrs", c);
800 }
801 }
802
803 if (!srcfilename || !dstfilename || (fReadFromStdIn && !filesize))
804 return errorSyntax(USAGE_CONVERTFROMRAW, "Incorrect number of parameters");
805 RTPrintf("Converting from raw image file=\"%s\" to file=\"%s\"...\n",
806 srcfilename, dstfilename);
807
808 PVBOXHDD pDisk = NULL;
809
810 PVDINTERFACE pVDIfs = NULL;
811 VDINTERFACE vdInterfaceError;
812 VDINTERFACEERROR vdInterfaceErrorCallbacks;
813 vdInterfaceErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
814 vdInterfaceErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
815 vdInterfaceErrorCallbacks.pfnError = handleVDError;
816 vdInterfaceErrorCallbacks.pfnMessage = NULL;
817
818 rc = VDInterfaceAdd(&vdInterfaceError, "VBoxManage_IError", VDINTERFACETYPE_ERROR,
819 &vdInterfaceErrorCallbacks, NULL, &pVDIfs);
820 AssertRC(rc);
821
822 /* open raw image file. */
823 RTFILE File;
824 if (fReadFromStdIn)
825 File = 0;
826 else
827 rc = RTFileOpen(&File, srcfilename, RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE);
828 if (RT_FAILURE(rc))
829 {
830 RTPrintf("File=\"%s\" open error: %Rrf\n", srcfilename, rc);
831 goto out;
832 }
833
834 uint64_t cbFile;
835 /* get image size. */
836 if (fReadFromStdIn)
837 cbFile = RTStrToUInt64(filesize);
838 else
839 rc = RTFileGetSize(File, &cbFile);
840 if (RT_FAILURE(rc))
841 {
842 RTPrintf("Error getting image size for file \"%s\": %Rrc\n", srcfilename, rc);
843 goto out;
844 }
845
846 RTPrintf("Creating %s image with size %RU64 bytes (%RU64MB)...\n", (uImageFlags & VD_IMAGE_FLAGS_FIXED) ? "fixed" : "dynamic", cbFile, (cbFile + _1M - 1) / _1M);
847 char pszComment[256];
848 RTStrPrintf(pszComment, sizeof(pszComment), "Converted image from %s", srcfilename);
849 rc = VDCreate(pVDIfs, &pDisk);
850 if (RT_FAILURE(rc))
851 {
852 RTPrintf("Error while creating the virtual disk container: %Rrc\n", rc);
853 goto out;
854 }
855
856 Assert(RT_MIN(cbFile / 512 / 16 / 63, 16383) -
857 (unsigned int)RT_MIN(cbFile / 512 / 16 / 63, 16383) == 0);
858 PDMMEDIAGEOMETRY PCHS, LCHS;
859 PCHS.cCylinders = (unsigned int)RT_MIN(cbFile / 512 / 16 / 63, 16383);
860 PCHS.cHeads = 16;
861 PCHS.cSectors = 63;
862 LCHS.cCylinders = 0;
863 LCHS.cHeads = 0;
864 LCHS.cSectors = 0;
865 rc = VDCreateBase(pDisk, format, dstfilename, cbFile,
866 uImageFlags, pszComment, &PCHS, &LCHS, NULL,
867 VD_OPEN_FLAGS_NORMAL, NULL, NULL);
868 if (RT_FAILURE(rc))
869 {
870 RTPrintf("Error while creating the disk image \"%s\": %Rrc\n", dstfilename, rc);
871 goto out;
872 }
873
874 size_t cbBuffer;
875 cbBuffer = _1M;
876 pvBuf = RTMemAlloc(cbBuffer);
877 if (!pvBuf)
878 {
879 rc = VERR_NO_MEMORY;
880 RTPrintf("Not enough memory allocating buffers for image \"%s\": %Rrc\n", dstfilename, rc);
881 goto out;
882 }
883
884 uint64_t offFile;
885 offFile = 0;
886 while (offFile < cbFile)
887 {
888 size_t cbRead;
889 size_t cbToRead;
890 cbRead = 0;
891 cbToRead = cbFile - offFile >= (uint64_t)cbBuffer ?
892 cbBuffer : (size_t) (cbFile - offFile);
893 rc = RTFileRead(File, pvBuf, cbToRead, &cbRead);
894 if (RT_FAILURE(rc) || !cbRead)
895 break;
896 rc = VDWrite(pDisk, offFile, pvBuf, cbRead);
897 if (RT_FAILURE(rc))
898 {
899 RTPrintf("Failed to write to disk image \"%s\": %Rrc\n", dstfilename, rc);
900 goto out;
901 }
902 offFile += cbRead;
903 }
904
905out:
906 if (pvBuf)
907 RTMemFree(pvBuf);
908 if (pDisk)
909 VDClose(pDisk, RT_FAILURE(rc));
910 if (File != NIL_RTFILE)
911 RTFileClose(File);
912
913 return RT_FAILURE(rc);
914}
915
916static const RTGETOPTDEF g_aAddiSCSIDiskOptions[] =
917{
918 { "--server", 's', RTGETOPT_REQ_STRING },
919 { "-server", 's', RTGETOPT_REQ_STRING }, // deprecated
920 { "--target", 'T', RTGETOPT_REQ_STRING },
921 { "-target", 'T', RTGETOPT_REQ_STRING }, // deprecated
922 { "--port", 'p', RTGETOPT_REQ_STRING },
923 { "-port", 'p', RTGETOPT_REQ_STRING }, // deprecated
924 { "--lun", 'l', RTGETOPT_REQ_STRING },
925 { "-lun", 'l', RTGETOPT_REQ_STRING }, // deprecated
926 { "--encodedlun", 'L', RTGETOPT_REQ_STRING },
927 { "-encodedlun", 'L', RTGETOPT_REQ_STRING }, // deprecated
928 { "--username", 'u', RTGETOPT_REQ_STRING },
929 { "-username", 'u', RTGETOPT_REQ_STRING }, // deprecated
930 { "--password", 'P', RTGETOPT_REQ_STRING },
931 { "-password", 'P', RTGETOPT_REQ_STRING }, // deprecated
932 { "--type", 't', RTGETOPT_REQ_STRING },
933 { "-type", 't', RTGETOPT_REQ_STRING }, // deprecated
934 { "--comment", 'c', RTGETOPT_REQ_STRING },
935 { "-comment", 'c', RTGETOPT_REQ_STRING }, // deprecated
936 { "--intnet", 'I', RTGETOPT_REQ_NOTHING },
937 { "-intnet", 'I', RTGETOPT_REQ_NOTHING }, // deprecated
938};
939
940int handleAddiSCSIDisk(HandlerArg *a)
941{
942 HRESULT rc;
943 int vrc;
944 Bstr server;
945 Bstr target;
946 Bstr port;
947 Bstr lun;
948 Bstr username;
949 Bstr password;
950 Bstr comment;
951 bool fIntNet = false;
952 MediumType_T DiskType = MediumType_Normal;
953
954 int c;
955 RTGETOPTUNION ValueUnion;
956 RTGETOPTSTATE GetState;
957 // start at 0 because main() has hacked both the argc and argv given to us
958 RTGetOptInit(&GetState, a->argc, a->argv, g_aAddiSCSIDiskOptions, RT_ELEMENTS(g_aAddiSCSIDiskOptions), 0, 0 /* fFlags */);
959 while ((c = RTGetOpt(&GetState, &ValueUnion)))
960 {
961 switch (c)
962 {
963 case 's': // --server
964 server = ValueUnion.psz;
965 break;
966
967 case 'T': // --target
968 target = ValueUnion.psz;
969 break;
970
971 case 'p': // --port
972 port = ValueUnion.psz;
973 break;
974
975 case 'l': // --lun
976 lun = ValueUnion.psz;
977 break;
978
979 case 'L': // --encodedlun
980 lun = BstrFmt("enc%s", ValueUnion.psz);
981 break;
982
983 case 'u': // --username
984 username = ValueUnion.psz;
985 break;
986
987 case 'P': // --password
988 password = ValueUnion.psz;
989 break;
990
991 case 't': // --type
992 vrc = parseDiskType(ValueUnion.psz, &DiskType);
993 if (RT_FAILURE(vrc))
994 return errorArgument("Invalid hard disk type '%s'", ValueUnion.psz);
995 break;
996
997 case 'c': // --comment
998 comment = ValueUnion.psz;
999 break;
1000
1001 case 'I': // --intnet
1002 fIntNet = true;
1003 break;
1004
1005 case VINF_GETOPT_NOT_OPTION:
1006 return errorSyntax(USAGE_ADDISCSIDISK, "Invalid parameter '%s'", ValueUnion.psz);
1007
1008 default:
1009 if (c > 0)
1010 {
1011 if (RT_C_IS_PRINT(c))
1012 return errorSyntax(USAGE_ADDISCSIDISK, "Invalid option -%c", c);
1013 else
1014 return errorSyntax(USAGE_ADDISCSIDISK, "Invalid option case %i", c);
1015 }
1016 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
1017 return errorSyntax(USAGE_ADDISCSIDISK, "unknown option: %s\n", ValueUnion.psz);
1018 else if (ValueUnion.pDef)
1019 return errorSyntax(USAGE_ADDISCSIDISK, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
1020 else
1021 return errorSyntax(USAGE_ADDISCSIDISK, "error: %Rrs", c);
1022 }
1023 }
1024
1025 /* check for required options */
1026 if (!server || !target)
1027 return errorSyntax(USAGE_ADDISCSIDISK, "Parameters --server and --target are required");
1028
1029 do
1030 {
1031 ComPtr<IMedium> hardDisk;
1032 /** @todo move the location stuff to Main, which can use pfnComposeName
1033 * from the disk backends to construct the location properly. Also do
1034 * not use slashes to separate the parts, as otherwise only the last
1035 * element comtaining information will be shown. */
1036 if (lun.isEmpty() || lun == "0" || lun == "enc0")
1037 {
1038 CHECK_ERROR_BREAK (a->virtualBox,
1039 CreateHardDisk(Bstr ("iSCSI"),
1040 BstrFmt ("%ls|%ls", server.raw(), target.raw()),
1041 hardDisk.asOutParam()));
1042 }
1043 else
1044 {
1045 CHECK_ERROR_BREAK (a->virtualBox,
1046 CreateHardDisk(Bstr ("iSCSI"),
1047 BstrFmt ("%ls|%ls|%ls", server.raw(), target.raw(), lun.raw()),
1048 hardDisk.asOutParam()));
1049 }
1050 CheckComRCBreakRC (rc);
1051
1052 if (!comment.isNull())
1053 CHECK_ERROR(hardDisk, COMSETTER(Description)(comment));
1054
1055 if (!port.isNull())
1056 server = BstrFmt ("%ls:%ls", server.raw(), port.raw());
1057
1058 com::SafeArray <BSTR> names;
1059 com::SafeArray <BSTR> values;
1060
1061 Bstr ("TargetAddress").detachTo (names.appendedRaw());
1062 server.detachTo (values.appendedRaw());
1063 Bstr ("TargetName").detachTo (names.appendedRaw());
1064 target.detachTo (values.appendedRaw());
1065
1066 if (!lun.isNull())
1067 {
1068 Bstr ("LUN").detachTo (names.appendedRaw());
1069 lun.detachTo (values.appendedRaw());
1070 }
1071 if (!username.isNull())
1072 {
1073 Bstr ("InitiatorUsername").detachTo (names.appendedRaw());
1074 username.detachTo (values.appendedRaw());
1075 }
1076 if (!password.isNull())
1077 {
1078 Bstr ("InitiatorSecret").detachTo (names.appendedRaw());
1079 password.detachTo (values.appendedRaw());
1080 }
1081
1082 /// @todo add --initiator option
1083 Bstr ("InitiatorName").detachTo (names.appendedRaw());
1084 Bstr ("iqn.2008-04.com.sun.virtualbox.initiator").detachTo (values.appendedRaw());
1085
1086 /// @todo add --targetName and --targetPassword options
1087
1088 if (fIntNet)
1089 {
1090 Bstr ("HostIPStack").detachTo (names.appendedRaw());
1091 Bstr ("0").detachTo (values.appendedRaw());
1092 }
1093
1094 CHECK_ERROR_BREAK (hardDisk,
1095 SetProperties (ComSafeArrayAsInParam (names),
1096 ComSafeArrayAsInParam (values)));
1097
1098 if (DiskType != MediumType_Normal)
1099 {
1100 CHECK_ERROR(hardDisk, COMSETTER(Type)(DiskType));
1101 }
1102
1103 Bstr guid;
1104 CHECK_ERROR(hardDisk, COMGETTER(Id)(guid.asOutParam()));
1105 RTPrintf("iSCSI disk created. UUID: %s\n", Utf8Str(guid).raw());
1106 }
1107 while (0);
1108
1109 return SUCCEEDED(rc) ? 0 : 1;
1110}
1111
1112static const RTGETOPTDEF g_aShowHardDiskInfoOptions[] =
1113{
1114 { "--dummy", 256, RTGETOPT_REQ_NOTHING }, // placeholder for C++
1115};
1116
1117int handleShowHardDiskInfo(HandlerArg *a)
1118{
1119 HRESULT rc;
1120 const char *FilenameOrUuid = NULL;
1121
1122 int c;
1123 RTGETOPTUNION ValueUnion;
1124 RTGETOPTSTATE GetState;
1125 // start at 0 because main() has hacked both the argc and argv given to us
1126 RTGetOptInit(&GetState, a->argc, a->argv, g_aShowHardDiskInfoOptions, RT_ELEMENTS(g_aShowHardDiskInfoOptions), 0, 0 /* fFlags */);
1127 while ((c = RTGetOpt(&GetState, &ValueUnion)))
1128 {
1129 switch (c)
1130 {
1131 case VINF_GETOPT_NOT_OPTION:
1132 if (!FilenameOrUuid)
1133 FilenameOrUuid = ValueUnion.psz;
1134 else
1135 return errorSyntax(USAGE_SHOWHDINFO, "Invalid parameter '%s'", ValueUnion.psz);
1136 break;
1137
1138 default:
1139 if (c > 0)
1140 {
1141 if (RT_C_IS_PRINT(c))
1142 return errorSyntax(USAGE_SHOWHDINFO, "Invalid option -%c", c);
1143 else
1144 return errorSyntax(USAGE_SHOWHDINFO, "Invalid option case %i", c);
1145 }
1146 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
1147 return errorSyntax(USAGE_SHOWHDINFO, "unknown option: %s\n", ValueUnion.psz);
1148 else if (ValueUnion.pDef)
1149 return errorSyntax(USAGE_SHOWHDINFO, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
1150 else
1151 return errorSyntax(USAGE_SHOWHDINFO, "error: %Rrs", c);
1152 }
1153 }
1154
1155 /* check for required options */
1156 if (!FilenameOrUuid)
1157 return errorSyntax(USAGE_SHOWHDINFO, "Disk name or UUID required");
1158
1159 ComPtr<IMedium> hardDisk;
1160 bool unknown = false;
1161 /* first guess is that it's a UUID */
1162 Bstr uuid(FilenameOrUuid);
1163 rc = a->virtualBox->GetHardDisk(uuid, hardDisk.asOutParam());
1164 /* no? then it must be a filename */
1165 if (FAILED (rc))
1166 {
1167 rc = a->virtualBox->FindHardDisk(Bstr(FilenameOrUuid), hardDisk.asOutParam());
1168 /* no? well, then it's an unkwnown image */
1169 if (FAILED (rc))
1170 {
1171 rc = a->virtualBox->OpenHardDisk(Bstr(FilenameOrUuid), AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), hardDisk.asOutParam());
1172 if (rc == VBOX_E_FILE_ERROR)
1173 {
1174 char szFilenameAbs[RTPATH_MAX] = "";
1175 int vrc = RTPathAbs(FilenameOrUuid, szFilenameAbs, sizeof(szFilenameAbs));
1176 if (RT_FAILURE(vrc))
1177 {
1178 RTPrintf("Cannot convert filename \"%s\" to absolute path\n", FilenameOrUuid);
1179 return 1;
1180 }
1181 CHECK_ERROR(a->virtualBox, OpenHardDisk(Bstr(szFilenameAbs), AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), hardDisk.asOutParam()));
1182 }
1183 else if (FAILED(rc))
1184 CHECK_ERROR(a->virtualBox, OpenHardDisk(Bstr(FilenameOrUuid), AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), hardDisk.asOutParam()));
1185 if (SUCCEEDED (rc))
1186 {
1187 unknown = true;
1188 }
1189 }
1190 }
1191 do
1192 {
1193 if (!SUCCEEDED(rc))
1194 break;
1195
1196 hardDisk->COMGETTER(Id)(uuid.asOutParam());
1197 RTPrintf("UUID: %s\n", Utf8Str(uuid).raw());
1198
1199 /* check for accessibility */
1200 /// @todo NEWMEDIA check accessibility of all parents
1201 /// @todo NEWMEDIA print the full state value
1202 MediumState_T state;
1203 CHECK_ERROR_BREAK (hardDisk, COMGETTER(State)(&state));
1204 RTPrintf("Accessible: %s\n", state != MediumState_Inaccessible ? "yes" : "no");
1205
1206 if (state == MediumState_Inaccessible)
1207 {
1208 Bstr err;
1209 CHECK_ERROR_BREAK (hardDisk, COMGETTER(LastAccessError)(err.asOutParam()));
1210 RTPrintf("Access Error: %lS\n", err.raw());
1211 }
1212
1213 Bstr description;
1214 hardDisk->COMGETTER(Description)(description.asOutParam());
1215 if (description)
1216 {
1217 RTPrintf("Description: %lS\n", description.raw());
1218 }
1219
1220 ULONG64 logicalSize;
1221 hardDisk->COMGETTER(LogicalSize)(&logicalSize);
1222 RTPrintf("Logical size: %llu MBytes\n", logicalSize);
1223 ULONG64 actualSize;
1224 hardDisk->COMGETTER(Size)(&actualSize);
1225 RTPrintf("Current size on disk: %llu MBytes\n", actualSize >> 20);
1226
1227 ComPtr <IMedium> parent;
1228 hardDisk->COMGETTER(Parent) (parent.asOutParam());
1229
1230 MediumType_T type;
1231 hardDisk->COMGETTER(Type)(&type);
1232 const char *typeStr = "unknown";
1233 switch (type)
1234 {
1235 case MediumType_Normal:
1236 if (!parent.isNull())
1237 typeStr = "normal (differencing)";
1238 else
1239 typeStr = "normal (base)";
1240 break;
1241 case MediumType_Immutable:
1242 typeStr = "immutable";
1243 break;
1244 case MediumType_Writethrough:
1245 typeStr = "writethrough";
1246 break;
1247 }
1248 RTPrintf("Type: %s\n", typeStr);
1249
1250 Bstr format;
1251 hardDisk->COMGETTER(Format)(format.asOutParam());
1252 RTPrintf("Storage format: %lS\n", format.raw());
1253
1254 if (!unknown)
1255 {
1256 com::SafeArray<BSTR> machineIds;
1257 hardDisk->COMGETTER(MachineIds)(ComSafeArrayAsOutParam(machineIds));
1258 for (size_t j = 0; j < machineIds.size(); ++ j)
1259 {
1260 ComPtr<IMachine> machine;
1261 CHECK_ERROR(a->virtualBox, GetMachine(machineIds[j], machine.asOutParam()));
1262 ASSERT(machine);
1263 Bstr name;
1264 machine->COMGETTER(Name)(name.asOutParam());
1265 machine->COMGETTER(Id)(uuid.asOutParam());
1266 RTPrintf("%s%lS (UUID: %lS)\n",
1267 j == 0 ? "In use by VMs: " : " ",
1268 name.raw(), machineIds[j]);
1269 }
1270 /// @todo NEWMEDIA check usage in snapshots too
1271 /// @todo NEWMEDIA also list children
1272 }
1273
1274 Bstr loc;
1275 hardDisk->COMGETTER(Location)(loc.asOutParam());
1276 RTPrintf("Location: %lS\n", loc.raw());
1277
1278 /* print out information specific for differencing hard disks */
1279 if (!parent.isNull())
1280 {
1281 BOOL autoReset = FALSE;
1282 hardDisk->COMGETTER(AutoReset)(&autoReset);
1283 RTPrintf("Auto-Reset: %s\n", autoReset ? "on" : "off");
1284 }
1285 }
1286 while (0);
1287
1288 if (unknown)
1289 {
1290 /* close the unknown hard disk to forget it again */
1291 hardDisk->Close();
1292 }
1293
1294 return SUCCEEDED(rc) ? 0 : 1;
1295}
1296
1297static const RTGETOPTDEF g_aOpenMediumOptions[] =
1298{
1299 { "disk", 'd', RTGETOPT_REQ_NOTHING },
1300 { "dvd", 'D', RTGETOPT_REQ_NOTHING },
1301 { "floppy", 'f', RTGETOPT_REQ_NOTHING },
1302 { "--type", 't', RTGETOPT_REQ_STRING },
1303 { "-type", 't', RTGETOPT_REQ_STRING }, // deprecated
1304 { "--uuid", 'u', RTGETOPT_REQ_UUID },
1305 { "--parentuuid", 'p', RTGETOPT_REQ_UUID },
1306};
1307
1308int handleOpenMedium(HandlerArg *a)
1309{
1310 HRESULT rc = S_OK;
1311 int vrc;
1312 enum {
1313 CMD_NONE,
1314 CMD_DISK,
1315 CMD_DVD,
1316 CMD_FLOPPY
1317 } cmd = CMD_NONE;
1318 const char *Filename = NULL;
1319 MediumType_T DiskType = MediumType_Normal;
1320 bool fDiskType = false;
1321 bool fSetImageId = false;
1322 bool fSetParentId = false;
1323 Guid ImageId;
1324 ImageId.clear();
1325 Guid ParentId;
1326 ParentId.clear();
1327
1328 int c;
1329 RTGETOPTUNION ValueUnion;
1330 RTGETOPTSTATE GetState;
1331 // start at 0 because main() has hacked both the argc and argv given to us
1332 RTGetOptInit(&GetState, a->argc, a->argv, g_aOpenMediumOptions, RT_ELEMENTS(g_aOpenMediumOptions), 0, 0 /* fFlags */);
1333 while ((c = RTGetOpt(&GetState, &ValueUnion)))
1334 {
1335 switch (c)
1336 {
1337 case 'd': // disk
1338 if (cmd != CMD_NONE)
1339 return errorSyntax(USAGE_OPENMEDIUM, "Only one command can be specified: '%s'", ValueUnion.psz);
1340 cmd = CMD_DISK;
1341 break;
1342
1343 case 'D': // DVD
1344 if (cmd != CMD_NONE)
1345 return errorSyntax(USAGE_OPENMEDIUM, "Only one command can be specified: '%s'", ValueUnion.psz);
1346 cmd = CMD_DVD;
1347 break;
1348
1349 case 'f': // floppy
1350 if (cmd != CMD_NONE)
1351 return errorSyntax(USAGE_OPENMEDIUM, "Only one command can be specified: '%s'", ValueUnion.psz);
1352 cmd = CMD_FLOPPY;
1353 break;
1354
1355 case 't': // --type
1356 vrc = parseDiskType(ValueUnion.psz, &DiskType);
1357 if (RT_FAILURE(vrc))
1358 return errorArgument("Invalid hard disk type '%s'", ValueUnion.psz);
1359 fDiskType = true;
1360 break;
1361
1362 case 'u': // --uuid
1363 ImageId = ValueUnion.Uuid;
1364 fSetImageId = true;
1365 break;
1366
1367 case 'p': // --parentuuid
1368 ParentId = ValueUnion.Uuid;
1369 fSetParentId = true;
1370 break;
1371
1372 case VINF_GETOPT_NOT_OPTION:
1373 if (!Filename)
1374 Filename = ValueUnion.psz;
1375 else
1376 return errorSyntax(USAGE_OPENMEDIUM, "Invalid parameter '%s'", ValueUnion.psz);
1377 break;
1378
1379 default:
1380 if (c > 0)
1381 {
1382 if (RT_C_IS_PRINT(c))
1383 return errorSyntax(USAGE_OPENMEDIUM, "Invalid option -%c", c);
1384 else
1385 return errorSyntax(USAGE_OPENMEDIUM, "Invalid option case %i", c);
1386 }
1387 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
1388 return errorSyntax(USAGE_OPENMEDIUM, "unknown option: %s\n", ValueUnion.psz);
1389 else if (ValueUnion.pDef)
1390 return errorSyntax(USAGE_OPENMEDIUM, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
1391 else
1392 return errorSyntax(USAGE_OPENMEDIUM, "error: %Rrs", c);
1393 }
1394 }
1395
1396 /* check for required options */
1397 if (cmd == CMD_NONE)
1398 return errorSyntax(USAGE_OPENMEDIUM, "Command variant disk/dvd/floppy required");
1399 if (!Filename)
1400 return errorSyntax(USAGE_OPENMEDIUM, "Disk name required");
1401
1402 /** @todo remove this hack!
1403 * First try opening the image as is (using the regular API semantics for
1404 * images with relative path or without path), and if that fails with a
1405 * file related error then try it again with what the client thinks the
1406 * relative path would mean. Requires doing the command twice in certain
1407 * cases. This is an ugly hack and needs to be removed whevever we have a
1408 * chance to clean up the API semantics. */
1409 if (cmd == CMD_DISK)
1410 {
1411 ComPtr<IMedium> hardDisk;
1412 Bstr ImageIdStr = BstrFmt("%RTuuid", &ImageId);
1413 Bstr ParentIdStr = BstrFmt("%RTuuid", &ParentId);
1414 rc = a->virtualBox->OpenHardDisk(Bstr(Filename), AccessMode_ReadWrite, fSetImageId, ImageIdStr, fSetParentId, ParentIdStr, hardDisk.asOutParam());
1415 if (rc == VBOX_E_FILE_ERROR)
1416 {
1417 char szFilenameAbs[RTPATH_MAX] = "";
1418 int vrc = RTPathAbs(Filename, szFilenameAbs, sizeof(szFilenameAbs));
1419 if (RT_FAILURE(vrc))
1420 {
1421 RTPrintf("Cannot convert filename \"%s\" to absolute path\n", Filename);
1422 return 1;
1423 }
1424 CHECK_ERROR(a->virtualBox, OpenHardDisk(Bstr(szFilenameAbs), AccessMode_ReadWrite, fSetImageId, ImageIdStr, fSetParentId, ParentIdStr, hardDisk.asOutParam()));
1425 }
1426 else if (FAILED(rc))
1427 CHECK_ERROR(a->virtualBox, OpenHardDisk(Bstr(Filename), AccessMode_ReadWrite, fSetImageId, ImageIdStr, fSetParentId, ParentIdStr, hardDisk.asOutParam()));
1428 if (SUCCEEDED(rc) && hardDisk)
1429 {
1430 /* change the type if requested */
1431 if (DiskType != MediumType_Normal)
1432 {
1433 CHECK_ERROR(hardDisk, COMSETTER(Type)(DiskType));
1434 }
1435 }
1436 }
1437 else if (cmd == CMD_DVD)
1438 {
1439 if (fDiskType || fSetImageId || fSetParentId)
1440 return errorSyntax(USAGE_OPENMEDIUM, "Invalid option for DVD images");
1441 ComPtr<IMedium> dvdImage;
1442 rc = a->virtualBox->OpenDVDImage(Bstr(Filename), Bstr(), dvdImage.asOutParam());
1443 if (rc == VBOX_E_FILE_ERROR)
1444 {
1445 char szFilenameAbs[RTPATH_MAX] = "";
1446 int vrc = RTPathAbs(Filename, szFilenameAbs, sizeof(szFilenameAbs));
1447 if (RT_FAILURE(vrc))
1448 {
1449 RTPrintf("Cannot convert filename \"%s\" to absolute path\n", Filename);
1450 return 1;
1451 }
1452 CHECK_ERROR(a->virtualBox, OpenDVDImage(Bstr(szFilenameAbs), Bstr(), dvdImage.asOutParam()));
1453 }
1454 else if (FAILED(rc))
1455 CHECK_ERROR(a->virtualBox, OpenDVDImage(Bstr(Filename), Bstr(), dvdImage.asOutParam()));
1456 }
1457 else if (cmd == CMD_FLOPPY)
1458 {
1459 if (fDiskType || fSetImageId || fSetParentId)
1460 return errorSyntax(USAGE_OPENMEDIUM, "Invalid option for floppy images");
1461 ComPtr<IMedium> floppyImage;
1462 rc = a->virtualBox->OpenFloppyImage(Bstr(Filename), Bstr(), floppyImage.asOutParam());
1463 if (rc == VBOX_E_FILE_ERROR)
1464 {
1465 char szFilenameAbs[RTPATH_MAX] = "";
1466 int vrc = RTPathAbs(Filename, szFilenameAbs, sizeof(szFilenameAbs));
1467 if (RT_FAILURE(vrc))
1468 {
1469 RTPrintf("Cannot convert filename \"%s\" to absolute path\n", Filename);
1470 return 1;
1471 }
1472 CHECK_ERROR(a->virtualBox, OpenFloppyImage(Bstr(szFilenameAbs), Bstr(), floppyImage.asOutParam()));
1473 }
1474 else if (FAILED(rc))
1475 CHECK_ERROR(a->virtualBox, OpenFloppyImage(Bstr(Filename), Bstr(), floppyImage.asOutParam()));
1476 }
1477
1478 return SUCCEEDED(rc) ? 0 : 1;
1479}
1480
1481static const RTGETOPTDEF g_aCloseMediumOptions[] =
1482{
1483 { "disk", 'd', RTGETOPT_REQ_NOTHING },
1484 { "dvd", 'D', RTGETOPT_REQ_NOTHING },
1485 { "floppy", 'f', RTGETOPT_REQ_NOTHING },
1486};
1487
1488int handleCloseMedium(HandlerArg *a)
1489{
1490 HRESULT rc = S_OK;
1491 enum {
1492 CMD_NONE,
1493 CMD_DISK,
1494 CMD_DVD,
1495 CMD_FLOPPY
1496 } cmd = CMD_NONE;
1497 const char *FilenameOrUuid = NULL;
1498
1499 int c;
1500 RTGETOPTUNION ValueUnion;
1501 RTGETOPTSTATE GetState;
1502 // start at 0 because main() has hacked both the argc and argv given to us
1503 RTGetOptInit(&GetState, a->argc, a->argv, g_aCloseMediumOptions, RT_ELEMENTS(g_aCloseMediumOptions), 0, 0 /* fFlags */);
1504 while ((c = RTGetOpt(&GetState, &ValueUnion)))
1505 {
1506 switch (c)
1507 {
1508 case 'd': // disk
1509 if (cmd != CMD_NONE)
1510 return errorSyntax(USAGE_CLOSEMEDIUM, "Only one command can be specified: '%s'", ValueUnion.psz);
1511 cmd = CMD_DISK;
1512 break;
1513
1514 case 'D': // DVD
1515 if (cmd != CMD_NONE)
1516 return errorSyntax(USAGE_CLOSEMEDIUM, "Only one command can be specified: '%s'", ValueUnion.psz);
1517 cmd = CMD_DVD;
1518 break;
1519
1520 case 'f': // floppy
1521 if (cmd != CMD_NONE)
1522 return errorSyntax(USAGE_CLOSEMEDIUM, "Only one command can be specified: '%s'", ValueUnion.psz);
1523 cmd = CMD_FLOPPY;
1524 break;
1525
1526 case VINF_GETOPT_NOT_OPTION:
1527 if (!FilenameOrUuid)
1528 FilenameOrUuid = ValueUnion.psz;
1529 else
1530 return errorSyntax(USAGE_CLOSEMEDIUM, "Invalid parameter '%s'", ValueUnion.psz);
1531 break;
1532
1533 default:
1534 if (c > 0)
1535 {
1536 if (RT_C_IS_PRINT(c))
1537 return errorSyntax(USAGE_CLOSEMEDIUM, "Invalid option -%c", c);
1538 else
1539 return errorSyntax(USAGE_CLOSEMEDIUM, "Invalid option case %i", c);
1540 }
1541 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
1542 return errorSyntax(USAGE_CLOSEMEDIUM, "unknown option: %s\n", ValueUnion.psz);
1543 else if (ValueUnion.pDef)
1544 return errorSyntax(USAGE_CLOSEMEDIUM, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
1545 else
1546 return errorSyntax(USAGE_CLOSEMEDIUM, "error: %Rrs", c);
1547 }
1548 }
1549
1550 /* check for required options */
1551 if (cmd == CMD_NONE)
1552 return errorSyntax(USAGE_CLOSEMEDIUM, "Command variant disk/dvd/floppy required");
1553 if (!FilenameOrUuid)
1554 return errorSyntax(USAGE_CLOSEMEDIUM, "Disk name or UUID required");
1555
1556 /* first guess is that it's a UUID */
1557 Bstr uuid(FilenameOrUuid);
1558
1559 if (cmd == CMD_DISK)
1560 {
1561 ComPtr<IMedium> hardDisk;
1562 rc = a->virtualBox->GetHardDisk(uuid, hardDisk.asOutParam());
1563 /* not a UUID or not registered? Then it must be a filename */
1564 if (!hardDisk)
1565 {
1566 CHECK_ERROR(a->virtualBox, FindHardDisk(Bstr(FilenameOrUuid), hardDisk.asOutParam()));
1567 }
1568 if (SUCCEEDED(rc) && hardDisk)
1569 {
1570 CHECK_ERROR(hardDisk, Close());
1571 }
1572 }
1573 else
1574 if (cmd == CMD_DVD)
1575 {
1576 ComPtr<IMedium> dvdImage;
1577 rc = a->virtualBox->GetDVDImage(uuid, dvdImage.asOutParam());
1578 /* not a UUID or not registered? Then it must be a filename */
1579 if (!dvdImage)
1580 {
1581 CHECK_ERROR(a->virtualBox, FindDVDImage(Bstr(FilenameOrUuid), dvdImage.asOutParam()));
1582 }
1583 if (SUCCEEDED(rc) && dvdImage)
1584 {
1585 CHECK_ERROR(dvdImage, Close());
1586 }
1587 }
1588 else
1589 if (cmd == CMD_FLOPPY)
1590 {
1591 ComPtr<IMedium> floppyImage;
1592 rc = a->virtualBox->GetFloppyImage(uuid, floppyImage.asOutParam());
1593 /* not a UUID or not registered? Then it must be a filename */
1594 if (!floppyImage)
1595 {
1596 CHECK_ERROR(a->virtualBox, FindFloppyImage(Bstr(FilenameOrUuid), floppyImage.asOutParam()));
1597 }
1598 if (SUCCEEDED(rc) && floppyImage)
1599 {
1600 CHECK_ERROR(floppyImage, Close());
1601 }
1602 }
1603
1604 return SUCCEEDED(rc) ? 0 : 1;
1605}
1606#endif /* !VBOX_ONLY_DOCS */
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