VirtualBox

source: vbox/trunk/src/VBox/Main/ApplianceImpl.cpp@ 21068

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

OVF: fix warnings and a wrong parameter possibly leading to bad config

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 209.2 KB
Line 
1/* $Id: ApplianceImpl.cpp 21068 2009-06-30 14:06:33Z vboxsync $ */
2/** @file
3 *
4 * IAppliance and IVirtualSystem COM class implementations.
5 */
6
7/*
8 * Copyright (C) 2008-2009 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 <iprt/stream.h>
24#include <iprt/path.h>
25#include <iprt/dir.h>
26#include <iprt/file.h>
27#include <iprt/s3.h>
28
29#include <VBox/param.h>
30#include <VBox/version.h>
31
32#include "ApplianceImpl.h"
33#include "VFSExplorerImpl.h"
34#include "VirtualBoxImpl.h"
35#include "GuestOSTypeImpl.h"
36#include "ProgressImpl.h"
37#include "MachineImpl.h"
38#include "HostNetworkInterfaceImpl.h"
39
40#include "Logging.h"
41
42#include "VBox/xml.h"
43
44using namespace std;
45
46////////////////////////////////////////////////////////////////////////////////
47//
48// hardware definitions
49//
50////////////////////////////////////////////////////////////////////////////////
51
52struct DiskImage
53{
54 Utf8Str strDiskId; // value from DiskSection/Disk/@diskId
55 int64_t iCapacity; // value from DiskSection/Disk/@capacity;
56 // (maximum size for dynamic images, I guess; we always translate this to bytes)
57 int64_t iPopulatedSize; // optional value from DiskSection/Disk/@populatedSize
58 // (actual used size of disk, always in bytes; can be an estimate of used disk
59 // space, but cannot be larger than iCapacity; -1 if not set)
60 Utf8Str strFormat; // value from DiskSection/Disk/@format
61 // typically http://www.vmware.com/specifications/vmdk.html#sparse
62
63 // fields from /References/File; the spec says the file reference from disk can be empty,
64 // so in that case, strFilename will be empty, then a new disk should be created
65 Utf8Str strHref; // value from /References/File/@href (filename); if empty, then the remaining fields are ignored
66 int64_t iSize; // value from /References/File/@size (optional according to spec; then we set -1 here)
67 int64_t iChunkSize; // value from /References/File/@chunkSize (optional, unsupported)
68 Utf8Str strCompression; // value from /References/File/@compression (optional, can be "gzip" according to spec)
69};
70
71struct VirtualHardwareItem
72{
73 Utf8Str strDescription;
74 Utf8Str strCaption;
75 Utf8Str strElementName;
76
77 uint32_t ulInstanceID;
78 uint32_t ulParent;
79
80 OVFResourceType_T resourceType;
81 Utf8Str strOtherResourceType;
82 Utf8Str strResourceSubType;
83
84 Utf8Str strHostResource; // "Abstractly specifies how a device shall connect to a resource on the deployment platform.
85 // Not all devices need a backing." Used with disk items, for which this references a virtual
86 // disk from the Disks section.
87 bool fAutomaticAllocation;
88 bool fAutomaticDeallocation;
89 Utf8Str strConnection; // "All Ethernet adapters that specify the same abstract network connection name within an OVF
90 // package shall be deployed on the same network. The abstract network connection name shall be
91 // listed in the NetworkSection at the outermost envelope level." We ignore this and only set up
92 // a network adapter depending on the network name.
93 Utf8Str strAddress; // "Device-specific. For an Ethernet adapter, this specifies the MAC address."
94 Utf8Str strAddressOnParent; // "For a device, this specifies its location on the controller."
95 Utf8Str strAllocationUnits; // "Specifies the units of allocation used. For example, “byte * 2^20”."
96 uint64_t ullVirtualQuantity; // "Specifies the quantity of resources presented. For example, “256”."
97 uint64_t ullReservation; // "Specifies the minimum quantity of resources guaranteed to be available."
98 uint64_t ullLimit; // "Specifies the maximum quantity of resources that will be granted."
99 uint64_t ullWeight; // "Specifies a relative priority for this allocation in relation to other allocations."
100
101 Utf8Str strConsumerVisibility;
102 Utf8Str strMappingBehavior;
103 Utf8Str strPoolID;
104 uint32_t ulBusNumber; // seen with IDE controllers, but not listed in OVF spec
105
106 uint32_t ulLineNumber; // line number of <Item> element in XML source; cached for error messages
107
108 VirtualHardwareItem()
109 : ulInstanceID(0), fAutomaticAllocation(false), fAutomaticDeallocation(false), ullVirtualQuantity(0), ullReservation(0), ullLimit(0), ullWeight(0), ulBusNumber(0), ulLineNumber(0)
110 {};
111};
112
113typedef map<Utf8Str, DiskImage> DiskImagesMap;
114
115struct VirtualSystem;
116
117typedef map<uint32_t, VirtualHardwareItem> HardwareItemsMap;
118
119struct HardDiskController
120{
121 uint32_t idController; // instance ID (Item/InstanceId); this gets referenced from HardDisk
122 enum ControllerSystemType { IDE, SATA, SCSI };
123 ControllerSystemType system; // one of IDE, SATA, SCSI
124 Utf8Str strControllerType; // controller subtype (Item/ResourceSubType); e.g. "LsiLogic"; can be empty (esp. for IDE)
125 Utf8Str strAddress; // for IDE
126 uint32_t ulBusNumber; // for IDE
127
128 HardDiskController()
129 : idController(0),
130 ulBusNumber(0)
131 {
132 }
133};
134
135typedef map<uint32_t, HardDiskController> ControllersMap;
136
137struct VirtualDisk
138{
139 uint32_t idController; // SCSI (or IDE) controller this disk is connected to;
140 // points into VirtualSystem.mapControllers
141 uint32_t ulAddressOnParent; // parsed strAddressOnParent of hardware item; will be 0 or 1 for IDE
142 // and possibly higher for disks attached to SCSI controllers (untested)
143 Utf8Str strDiskId; // if the hard disk has an ovf:/disk/<id> reference,
144 // this receives the <id> component; points to one of the
145 // references in Appliance::Data.mapDisks
146};
147
148typedef map<Utf8Str, VirtualDisk> VirtualDisksMap;
149
150struct EthernetAdapter
151{
152 Utf8Str strAdapterType; // "PCNet32" or "E1000" or whatever; from <rasd:ResourceSubType>
153 Utf8Str strNetworkName; // from <rasd:Connection>
154};
155
156typedef list<EthernetAdapter> EthernetAdaptersList;
157
158struct VirtualSystem
159{
160 Utf8Str strName; // copy of VirtualSystem/@id
161
162 Utf8Str strDescription; // copy of VirtualSystem/Info content
163
164 CIMOSType_T cimos;
165 Utf8Str strCimosDesc; // readable description of the cimos type in the case of cimos = 0/1/102
166 Utf8Str strVirtualSystemType; // generic hardware description; OVF says this can be something like "vmx-4" or "xen";
167 // VMware Workstation 6.5 is "vmx-07"
168
169 HardwareItemsMap mapHardwareItems; // map of virtual hardware items, sorted by unique instance ID
170
171 uint64_t ullMemorySize; // always in bytes, copied from llHardwareItems; default = 0 (unspecified)
172 uint16_t cCPUs; // no. of CPUs, copied from llHardwareItems; default = 1
173
174 EthernetAdaptersList llEthernetAdapters; // (one for each VirtualSystem/Item[@ResourceType=10]element)
175
176 ControllersMap mapControllers;
177 // list of hard disk controllers
178 // (one for each VirtualSystem/Item[@ResourceType=6] element with accumulated data from children)
179
180 VirtualDisksMap mapVirtualDisks;
181 // (one for each VirtualSystem/Item[@ResourceType=17] element with accumulated data from children)
182
183 bool fHasFloppyDrive; // true if there's a floppy item in mapHardwareItems
184 bool fHasCdromDrive; // true if there's a CD-ROM item in mapHardwareItems; ISO images are not yet supported by OVFtool
185 bool fHasUsbController; // true if there's a USB controller item in mapHardwareItems
186
187 Utf8Str strSoundCardType; // if not empty, then the system wants a soundcard; this then specifies the hardware;
188 // VMware Workstation 6.5 uses "ensoniq1371" for example
189
190 Utf8Str strLicenseText; // license info if any; receives contents of VirtualSystem/EulaSection/License
191
192 Utf8Str strProduct; // product info if any; receives contents of VirtualSystem/ProductSection/Product
193 Utf8Str strVendor; // product info if any; receives contents of VirtualSystem/ProductSection/Vendor
194 Utf8Str strVersion; // product info if any; receives contents of VirtualSystem/ProductSection/Version
195 Utf8Str strProductUrl; // product info if any; receives contents of VirtualSystem/ProductSection/ProductUrl
196 Utf8Str strVendorUrl; // product info if any; receives contents of VirtualSystem/ProductSection/VendorUrl
197
198 VirtualSystem()
199 : ullMemorySize(0), cCPUs(1), fHasFloppyDrive(false), fHasCdromDrive(false), fHasUsbController(false)
200 {
201 }
202};
203
204////////////////////////////////////////////////////////////////////////////////
205//
206// Appliance data definition
207//
208////////////////////////////////////////////////////////////////////////////////
209
210// opaque private instance data of Appliance class
211struct Appliance::Data
212{
213 Utf8Str strPath; // file name last given to either read() or write()
214
215 DiskImagesMap mapDisks; // map of DiskImage structs, sorted by DiskImage.strDiskId
216
217 list<VirtualSystem> llVirtualSystems; // list of virtual systems, created by and valid after read()
218
219 list< ComObjPtr<VirtualSystemDescription> > virtualSystemDescriptions; //
220
221 list<Utf8Str> llWarnings;
222
223 ULONG ulWeightPerOperation; // for progress calculations
224};
225
226struct VirtualSystemDescription::Data
227{
228 list<VirtualSystemDescriptionEntry> llDescriptions;
229};
230
231////////////////////////////////////////////////////////////////////////////////
232//
233// internal helpers
234//
235////////////////////////////////////////////////////////////////////////////////
236
237static Utf8Str stripFilename(const Utf8Str &strFile)
238{
239 Utf8Str str2(strFile);
240 RTPathStripFilename(str2.mutableRaw());
241 return str2;
242}
243
244static const struct
245{
246 CIMOSType_T cim;
247 const char *pcszVbox;
248}
249 g_osTypes[] =
250 {
251 { CIMOSType_CIMOS_Unknown, SchemaDefs_OSTypeId_Other },
252 { CIMOSType_CIMOS_OS2, SchemaDefs_OSTypeId_OS2 },
253 { CIMOSType_CIMOS_MSDOS, SchemaDefs_OSTypeId_DOS },
254 { CIMOSType_CIMOS_WIN3x, SchemaDefs_OSTypeId_Windows31 },
255 { CIMOSType_CIMOS_WIN95, SchemaDefs_OSTypeId_Windows95 },
256 { CIMOSType_CIMOS_WIN98, SchemaDefs_OSTypeId_Windows98 },
257 { CIMOSType_CIMOS_WINNT, SchemaDefs_OSTypeId_WindowsNT4 },
258 { CIMOSType_CIMOS_NetWare, SchemaDefs_OSTypeId_Netware },
259 { CIMOSType_CIMOS_NovellOES, SchemaDefs_OSTypeId_Netware },
260 { CIMOSType_CIMOS_Solaris, SchemaDefs_OSTypeId_OpenSolaris },
261 { CIMOSType_CIMOS_SunOS, SchemaDefs_OSTypeId_OpenSolaris },
262 { CIMOSType_CIMOS_FreeBSD, SchemaDefs_OSTypeId_FreeBSD },
263 { CIMOSType_CIMOS_NetBSD, SchemaDefs_OSTypeId_NetBSD },
264 { CIMOSType_CIMOS_QNX, SchemaDefs_OSTypeId_QNX },
265 { CIMOSType_CIMOS_Windows2000, SchemaDefs_OSTypeId_Windows2000 },
266 { CIMOSType_CIMOS_WindowsMe, SchemaDefs_OSTypeId_WindowsMe },
267 { CIMOSType_CIMOS_OpenBSD, SchemaDefs_OSTypeId_OpenBSD },
268 { CIMOSType_CIMOS_WindowsXP, SchemaDefs_OSTypeId_WindowsXP },
269 { CIMOSType_CIMOS_WindowsXPEmbedded, SchemaDefs_OSTypeId_WindowsXP },
270 { CIMOSType_CIMOS_WindowsEmbeddedforPointofService, SchemaDefs_OSTypeId_WindowsXP },
271 { CIMOSType_CIMOS_MicrosoftWindowsServer2003, SchemaDefs_OSTypeId_Windows2003 },
272 { CIMOSType_CIMOS_MicrosoftWindowsServer2003_64, SchemaDefs_OSTypeId_Windows2003_64 },
273 { CIMOSType_CIMOS_WindowsXP_64, SchemaDefs_OSTypeId_WindowsXP_64 },
274 { CIMOSType_CIMOS_WindowsVista, SchemaDefs_OSTypeId_WindowsVista },
275 { CIMOSType_CIMOS_WindowsVista_64, SchemaDefs_OSTypeId_WindowsVista_64 },
276 { CIMOSType_CIMOS_MicrosoftWindowsServer2008, SchemaDefs_OSTypeId_Windows2008 },
277 { CIMOSType_CIMOS_MicrosoftWindowsServer2008_64, SchemaDefs_OSTypeId_Windows2008_64 },
278 { CIMOSType_CIMOS_FreeBSD_64, SchemaDefs_OSTypeId_FreeBSD_64 },
279 { CIMOSType_CIMOS_RedHatEnterpriseLinux, SchemaDefs_OSTypeId_RedHat },
280 { CIMOSType_CIMOS_RedHatEnterpriseLinux_64, SchemaDefs_OSTypeId_RedHat_64 },
281 { CIMOSType_CIMOS_Solaris_64, SchemaDefs_OSTypeId_OpenSolaris_64 },
282 { CIMOSType_CIMOS_SUSE, SchemaDefs_OSTypeId_OpenSUSE },
283 { CIMOSType_CIMOS_SLES, SchemaDefs_OSTypeId_OpenSUSE },
284 { CIMOSType_CIMOS_NovellLinuxDesktop, SchemaDefs_OSTypeId_OpenSUSE },
285 { CIMOSType_CIMOS_SUSE_64, SchemaDefs_OSTypeId_OpenSUSE_64 },
286 { CIMOSType_CIMOS_SLES_64, SchemaDefs_OSTypeId_OpenSUSE_64 },
287 { CIMOSType_CIMOS_LINUX, SchemaDefs_OSTypeId_Linux },
288 { CIMOSType_CIMOS_SunJavaDesktopSystem, SchemaDefs_OSTypeId_Linux },
289 { CIMOSType_CIMOS_TurboLinux, SchemaDefs_OSTypeId_Linux},
290
291 // { CIMOSType_CIMOS_TurboLinux_64, },
292
293 { CIMOSType_CIMOS_Mandriva, SchemaDefs_OSTypeId_Mandriva },
294 { CIMOSType_CIMOS_Mandriva_64, SchemaDefs_OSTypeId_Mandriva_64 },
295 { CIMOSType_CIMOS_Ubuntu, SchemaDefs_OSTypeId_Ubuntu },
296 { CIMOSType_CIMOS_Ubuntu_64, SchemaDefs_OSTypeId_Ubuntu_64 },
297 { CIMOSType_CIMOS_Debian, SchemaDefs_OSTypeId_Debian },
298 { CIMOSType_CIMOS_Debian_64, SchemaDefs_OSTypeId_Debian_64 },
299 { CIMOSType_CIMOS_Linux_2_4_x, SchemaDefs_OSTypeId_Linux24 },
300 { CIMOSType_CIMOS_Linux_2_4_x_64, SchemaDefs_OSTypeId_Linux24_64 },
301 { CIMOSType_CIMOS_Linux_2_6_x, SchemaDefs_OSTypeId_Linux26 },
302 { CIMOSType_CIMOS_Linux_2_6_x_64, SchemaDefs_OSTypeId_Linux26_64 },
303 { CIMOSType_CIMOS_Linux_64, SchemaDefs_OSTypeId_Linux26_64 }
304};
305
306/* Pattern structure for matching the os type description field */
307struct osTypePattern
308{
309 const char *pcszPattern;
310 const char *pcszVbox;
311};
312
313/* These are the 32-Bit ones. They are sorted by priority. */
314static const osTypePattern g_osTypesPattern[] =
315{
316 {"Windows NT", SchemaDefs_OSTypeId_WindowsNT4},
317 {"Windows XP", SchemaDefs_OSTypeId_WindowsXP},
318 {"Windows 2000", SchemaDefs_OSTypeId_Windows2000},
319 {"Windows 2003", SchemaDefs_OSTypeId_Windows2003},
320 {"Windows Vista", SchemaDefs_OSTypeId_WindowsVista},
321 {"Windows 2008", SchemaDefs_OSTypeId_Windows2008},
322 {"SUSE", SchemaDefs_OSTypeId_OpenSUSE},
323 {"Novell", SchemaDefs_OSTypeId_OpenSUSE},
324 {"Red Hat", SchemaDefs_OSTypeId_RedHat},
325 {"Mandriva", SchemaDefs_OSTypeId_Mandriva},
326 {"Ubuntu", SchemaDefs_OSTypeId_Ubuntu},
327 {"Debian", SchemaDefs_OSTypeId_Debian},
328 {"QNX", SchemaDefs_OSTypeId_QNX},
329 {"Linux 2.4", SchemaDefs_OSTypeId_Linux24},
330 {"Linux 2.6", SchemaDefs_OSTypeId_Linux26},
331 {"Linux", SchemaDefs_OSTypeId_Linux},
332 {"OpenSolaris", SchemaDefs_OSTypeId_OpenSolaris},
333 {"Solaris", SchemaDefs_OSTypeId_OpenSolaris},
334 {"FreeBSD", SchemaDefs_OSTypeId_FreeBSD},
335 {"NetBSD", SchemaDefs_OSTypeId_NetBSD},
336 {"Windows 95", SchemaDefs_OSTypeId_Windows95},
337 {"Windows 98", SchemaDefs_OSTypeId_Windows98},
338 {"Windows Me", SchemaDefs_OSTypeId_WindowsMe},
339 {"Windows 3.", SchemaDefs_OSTypeId_Windows31},
340 {"DOS", SchemaDefs_OSTypeId_DOS},
341 {"OS2", SchemaDefs_OSTypeId_OS2}
342};
343
344/* These are the 64-Bit ones. They are sorted by priority. */
345static const osTypePattern g_osTypesPattern64[] =
346{
347 {"Windows XP", SchemaDefs_OSTypeId_WindowsXP_64},
348 {"Windows 2003", SchemaDefs_OSTypeId_Windows2003_64},
349 {"Windows Vista", SchemaDefs_OSTypeId_WindowsVista_64},
350 {"Windows 2008", SchemaDefs_OSTypeId_Windows2008_64},
351 {"SUSE", SchemaDefs_OSTypeId_OpenSUSE_64},
352 {"Novell", SchemaDefs_OSTypeId_OpenSUSE_64},
353 {"Red Hat", SchemaDefs_OSTypeId_RedHat_64},
354 {"Mandriva", SchemaDefs_OSTypeId_Mandriva_64},
355 {"Ubuntu", SchemaDefs_OSTypeId_Ubuntu_64},
356 {"Debian", SchemaDefs_OSTypeId_Debian_64},
357 {"Linux 2.4", SchemaDefs_OSTypeId_Linux24_64},
358 {"Linux 2.6", SchemaDefs_OSTypeId_Linux26_64},
359 {"Linux", SchemaDefs_OSTypeId_Linux26_64},
360 {"OpenSolaris", SchemaDefs_OSTypeId_OpenSolaris_64},
361 {"Solaris", SchemaDefs_OSTypeId_OpenSolaris_64},
362 {"FreeBSD", SchemaDefs_OSTypeId_FreeBSD_64},
363};
364
365/**
366 * Private helper func that suggests a VirtualBox guest OS type
367 * for the given OVF operating system type.
368 * @param osTypeVBox
369 * @param c
370 * @param cStr
371 */
372static void convertCIMOSType2VBoxOSType(Utf8Str &strType, CIMOSType_T c, const Utf8Str &cStr)
373{
374 /* First check if the type is other/other_64 */
375 if (c == CIMOSType_CIMOS_Other)
376 {
377 for (size_t i=0; i < RT_ELEMENTS(g_osTypesPattern); ++i)
378 if (cStr.contains (g_osTypesPattern[i].pcszPattern, Utf8Str::CaseInsensitive))
379 {
380 strType = g_osTypesPattern[i].pcszVbox;
381 return;
382 }
383 }
384 else if (c == CIMOSType_CIMOS_Other_64)
385 {
386 for (size_t i=0; i < RT_ELEMENTS(g_osTypesPattern64); ++i)
387 if (cStr.contains (g_osTypesPattern64[i].pcszPattern, Utf8Str::CaseInsensitive))
388 {
389 strType = g_osTypesPattern64[i].pcszVbox;
390 return;
391 }
392 }
393
394 for (size_t i = 0; i < RT_ELEMENTS(g_osTypes); ++i)
395 {
396 if (c == g_osTypes[i].cim)
397 {
398 strType = g_osTypes[i].pcszVbox;
399 return;
400 }
401 }
402
403 strType = SchemaDefs_OSTypeId_Other;
404}
405
406/**
407 * Private helper func that suggests a VirtualBox guest OS type
408 * for the given OVF operating system type.
409 * @param osTypeVBox
410 * @param c
411 */
412static CIMOSType_T convertVBoxOSType2CIMOSType(const char *pcszVbox)
413{
414 for (size_t i = 0; i < RT_ELEMENTS(g_osTypes); ++i)
415 {
416 if (!RTStrICmp(pcszVbox, g_osTypes[i].pcszVbox))
417 return g_osTypes[i].cim;
418 }
419
420 return CIMOSType_CIMOS_Other;
421}
422
423////////////////////////////////////////////////////////////////////////////////
424//
425// IVirtualBox public methods
426//
427////////////////////////////////////////////////////////////////////////////////
428
429// This code is here so we won't have to include the appliance headers in the
430// IVirtualBox implementation.
431
432/**
433 * Implementation for IVirtualBox::createAppliance.
434 *
435 * @param anAppliance IAppliance object created if S_OK is returned.
436 * @return S_OK or error.
437 */
438STDMETHODIMP VirtualBox::CreateAppliance(IAppliance** anAppliance)
439{
440 HRESULT rc;
441
442 ComObjPtr<Appliance> appliance;
443 appliance.createObject();
444 rc = appliance->init(this);
445
446 if (SUCCEEDED(rc))
447 appliance.queryInterfaceTo(anAppliance);
448
449 return rc;
450}
451
452////////////////////////////////////////////////////////////////////////////////
453//
454// Appliance constructor / destructor
455//
456////////////////////////////////////////////////////////////////////////////////
457
458DEFINE_EMPTY_CTOR_DTOR(Appliance)
459struct shutup {};
460
461/**
462 * Appliance COM initializer.
463 * @param
464 * @return
465 */
466HRESULT Appliance::init(VirtualBox *aVirtualBox)
467{
468 /* Enclose the state transition NotReady->InInit->Ready */
469 AutoInitSpan autoInitSpan(this);
470 AssertReturn(autoInitSpan.isOk(), E_FAIL);
471
472 /* Weak reference to a VirtualBox object */
473 unconst(mVirtualBox) = aVirtualBox;
474
475 // initialize data
476 m = new Data;
477
478 /* Confirm a successful initialization */
479 autoInitSpan.setSucceeded();
480
481 return S_OK;
482}
483
484/**
485 * Appliance COM uninitializer.
486 * @return
487 */
488void Appliance::uninit()
489{
490 delete m;
491 m = NULL;
492}
493
494////////////////////////////////////////////////////////////////////////////////
495//
496// Appliance private methods
497//
498////////////////////////////////////////////////////////////////////////////////
499
500/**
501 * Private helper method that goes thru the elements of the given "current" element in the OVF XML
502 * and handles the contained child elements (which can be "Section" or "Content" elements).
503 *
504 * @param pcszPath Path spec of the XML file, for error messages.
505 * @param pReferencesElement "References" element from OVF, for looking up file specifications; can be NULL if no such element is present.
506 * @param pCurElem Element whose children are to be analyzed here.
507 * @return
508 */
509HRESULT Appliance::LoopThruSections(const char *pcszPath,
510 const xml::ElementNode *pReferencesElem,
511 const xml::ElementNode *pCurElem)
512{
513 HRESULT rc;
514
515 xml::NodesLoop loopChildren(*pCurElem);
516 const xml::ElementNode *pElem;
517 while ((pElem = loopChildren.forAllNodes()))
518 {
519 const char *pcszElemName = pElem->getName();
520 const char *pcszTypeAttr = "";
521 const xml::AttributeNode *pTypeAttr;
522 if ((pTypeAttr = pElem->findAttribute("type")))
523 pcszTypeAttr = pTypeAttr->getValue();
524
525 if ( (!strcmp(pcszElemName, "DiskSection"))
526 || ( (!strcmp(pcszElemName, "Section"))
527 && (!strcmp(pcszTypeAttr, "ovf:DiskSection_Type"))
528 )
529 )
530 {
531 if (!(SUCCEEDED((rc = HandleDiskSection(pcszPath, pReferencesElem, pElem)))))
532 return rc;
533 }
534 else if ( (!strcmp(pcszElemName, "NetworkSection"))
535 || ( (!strcmp(pcszElemName, "Section"))
536 && (!strcmp(pcszTypeAttr, "ovf:NetworkSection_Type"))
537 )
538 )
539 {
540 if (!(SUCCEEDED((rc = HandleNetworkSection(pcszPath, pElem)))))
541 return rc;
542 }
543 else if ( (!strcmp(pcszElemName, "DeploymentOptionSection")))
544 {
545 // TODO
546 }
547 else if ( (!strcmp(pcszElemName, "Info")))
548 {
549 // child of VirtualSystemCollection -- TODO
550 }
551 else if ( (!strcmp(pcszElemName, "ResourceAllocationSection")))
552 {
553 // child of VirtualSystemCollection -- TODO
554 }
555 else if ( (!strcmp(pcszElemName, "StartupSection")))
556 {
557 // child of VirtualSystemCollection -- TODO
558 }
559 else if ( (!strcmp(pcszElemName, "VirtualSystem"))
560 || ( (!strcmp(pcszElemName, "Content"))
561 && (!strcmp(pcszTypeAttr, "ovf:VirtualSystem_Type"))
562 )
563 )
564 {
565 if (!(SUCCEEDED((rc = HandleVirtualSystemContent(pcszPath, pElem)))))
566 return rc;
567 }
568 else if ( (!strcmp(pcszElemName, "VirtualSystemCollection"))
569 || ( (!strcmp(pcszElemName, "Content"))
570 && (!strcmp(pcszTypeAttr, "ovf:VirtualSystemCollection_Type"))
571 )
572 )
573 {
574 // TODO ResourceAllocationSection
575
576 // recurse for this, since it has VirtualSystem elements as children
577 if (!(SUCCEEDED((rc = LoopThruSections(pcszPath, pReferencesElem, pElem)))))
578 return rc;
579 }
580 }
581
582 return S_OK;
583}
584
585/**
586 * Private helper method that handles disk sections in the OVF XML.
587 * Gets called indirectly from IAppliance::read().
588 *
589 * @param pcszPath Path spec of the XML file, for error messages.
590 * @param pReferencesElement "References" element from OVF, for looking up file specifications; can be NULL if no such element is present.
591 * @param pSectionElem Section element for which this helper is getting called.
592 * @return
593 */
594HRESULT Appliance::HandleDiskSection(const char *pcszPath,
595 const xml::ElementNode *pReferencesElem,
596 const xml::ElementNode *pSectionElem)
597{
598 // contains "Disk" child elements
599 xml::NodesLoop loopDisks(*pSectionElem, "Disk");
600 const xml::ElementNode *pelmDisk;
601 while ((pelmDisk = loopDisks.forAllNodes()))
602 {
603 DiskImage d;
604 const char *pcszBad = NULL;
605 const char *pcszDiskId;
606 const char *pcszFormat;
607 if (!(pelmDisk->getAttributeValue("diskId", pcszDiskId)))
608 pcszBad = "diskId";
609 else if (!(pelmDisk->getAttributeValue("format", pcszFormat)))
610 pcszBad = "format";
611 else if (!(pelmDisk->getAttributeValue("capacity", d.iCapacity)))
612 pcszBad = "capacity";
613 else
614 {
615 d.strDiskId = pcszDiskId;
616 d.strFormat = pcszFormat;
617
618 if (!(pelmDisk->getAttributeValue("populatedSize", d.iPopulatedSize)))
619 // optional
620 d.iPopulatedSize = -1;
621
622 const char *pcszFileRef;
623 if (pelmDisk->getAttributeValue("fileRef", pcszFileRef)) // optional
624 {
625 // look up corresponding /References/File nodes (list built above)
626 const xml::ElementNode *pFileElem;
627 if ( pReferencesElem
628 && ((pFileElem = pReferencesElem->findChildElementFromId(pcszFileRef)))
629 )
630 {
631 // copy remaining values from file node then
632 const char *pcszBadInFile = NULL;
633 const char *pcszHref;
634 if (!(pFileElem->getAttributeValue("href", pcszHref)))
635 pcszBadInFile = "href";
636 else if (!(pFileElem->getAttributeValue("size", d.iSize)))
637 d.iSize = -1; // optional
638
639 d.strHref = pcszHref;
640
641 // if (!(pFileElem->getAttributeValue("size", d.iChunkSize))) TODO
642 d.iChunkSize = -1; // optional
643 const char *pcszCompression;
644 if (pFileElem->getAttributeValue("compression", pcszCompression))
645 d.strCompression = pcszCompression;
646
647 if (pcszBadInFile)
648 return setError(VBOX_E_FILE_ERROR,
649 tr("Error reading \"%s\": missing or invalid attribute '%s' in 'File' element, line %d"),
650 pcszPath,
651 pcszBadInFile,
652 pFileElem->getLineNumber());
653 }
654 else
655 return setError(VBOX_E_FILE_ERROR,
656 tr("Error reading \"%s\": cannot find References/File element for ID '%s' referenced by 'Disk' element, line %d"),
657 pcszPath,
658 pcszFileRef,
659 pelmDisk->getLineNumber());
660 }
661 }
662
663 if (pcszBad)
664 return setError(VBOX_E_FILE_ERROR,
665 tr("Error reading \"%s\": missing or invalid attribute '%s' in 'DiskSection' element, line %d"),
666 pcszPath,
667 pcszBad,
668 pelmDisk->getLineNumber());
669
670 m->mapDisks[d.strDiskId] = d;
671 }
672
673 return S_OK;
674}
675
676/**
677 * Private helper method that handles network sections in the OVF XML.
678 * Gets called indirectly from IAppliance::read().
679 *
680 * @param pcszPath Path spec of the XML file, for error messages.
681 * @param pSectionElem Section element for which this helper is getting called.
682 * @return
683 */
684HRESULT Appliance::HandleNetworkSection(const char * /* pcszPath */,
685 const xml::ElementNode * /* pSectionElem */)
686{
687 // we ignore network sections for now
688
689// xml::NodesLoop loopNetworks(*pSectionElem, "Network");
690// const xml::Node *pelmNetwork;
691// while ((pelmNetwork = loopNetworks.forAllNodes()))
692// {
693// Network n;
694// if (!(pelmNetwork->getAttributeValue("name", n.strNetworkName)))
695// return setError(VBOX_E_FILE_ERROR,
696// tr("Error reading \"%s\": missing 'name' attribute in 'Network', line %d"),
697// pcszPath,
698// pelmNetwork->getLineNumber());
699//
700// m->mapNetworks[n.strNetworkName] = n;
701// }
702
703 return S_OK;
704}
705
706/**
707 * Private helper method that handles a "VirtualSystem" element in the OVF XML.
708 * Gets called indirectly from IAppliance::read().
709 *
710 * @param pcszPath
711 * @param pContentElem
712 * @return
713 */
714HRESULT Appliance::HandleVirtualSystemContent(const char *pcszPath,
715 const xml::ElementNode *pelmVirtualSystem)
716{
717 VirtualSystem vsys;
718
719 const xml::AttributeNode *pIdAttr = pelmVirtualSystem->findAttribute("id");
720 if (pIdAttr)
721 vsys.strName = pIdAttr->getValue();
722
723 xml::NodesLoop loop(*pelmVirtualSystem); // all child elements
724 const xml::ElementNode *pelmThis;
725 while ((pelmThis = loop.forAllNodes()))
726 {
727 const char *pcszElemName = pelmThis->getName();
728 const xml::AttributeNode *pTypeAttr = pelmThis->findAttribute("type");
729 const char *pcszTypeAttr = (pTypeAttr) ? pTypeAttr->getValue() : "";
730
731 if ( (!strcmp(pcszElemName, "EulaSection"))
732 || (!strcmp(pcszTypeAttr, "ovf:EulaSection_Type"))
733 )
734 {
735 /* <EulaSection>
736 <Info ovf:msgid="6">License agreement for the Virtual System.</Info>
737 <License ovf:msgid="1">License terms can go in here.</License>
738 </EulaSection> */
739
740 const xml::ElementNode *pelmLicense;
741 if ((pelmLicense = pelmThis->findChildElement("License")))
742 vsys.strLicenseText = pelmLicense->getValue();
743 }
744 if ( (!strcmp(pcszElemName, "ProductSection"))
745 || (!strcmp(pcszTypeAttr, "ovf:ProductSection_Type"))
746 )
747 {
748 /* <Section ovf:required="false" xsi:type="ovf:ProductSection_Type">
749 <Info>Meta-information about the installed software</Info>
750 <Product>VAtest</Product>
751 <Vendor>SUN Microsystems</Vendor>
752 <Version>10.0</Version>
753 <ProductUrl>http://blogs.sun.com/VirtualGuru</ProductUrl>
754 <VendorUrl>http://www.sun.com</VendorUrl>
755 </Section> */
756 const xml::ElementNode *pelmProduct;
757 if ((pelmProduct = pelmThis->findChildElement("Product")))
758 vsys.strProduct = pelmProduct->getValue();
759 const xml::ElementNode *pelmVendor;
760 if ((pelmVendor = pelmThis->findChildElement("Vendor")))
761 vsys.strVendor = pelmVendor->getValue();
762 const xml::ElementNode *pelmVersion;
763 if ((pelmVersion = pelmThis->findChildElement("Version")))
764 vsys.strVersion = pelmVersion->getValue();
765 const xml::ElementNode *pelmProductUrl;
766 if ((pelmProductUrl = pelmThis->findChildElement("ProductUrl")))
767 vsys.strProductUrl = pelmProductUrl->getValue();
768 const xml::ElementNode *pelmVendorUrl;
769 if ((pelmVendorUrl = pelmThis->findChildElement("VendorUrl")))
770 vsys.strVendorUrl = pelmVendorUrl->getValue();
771 }
772 else if ( (!strcmp(pcszElemName, "VirtualHardwareSection"))
773 || (!strcmp(pcszTypeAttr, "ovf:VirtualHardwareSection_Type"))
774 )
775 {
776 const xml::ElementNode *pelmSystem, *pelmVirtualSystemType;
777 if ((pelmSystem = pelmThis->findChildElement("System")))
778 {
779 /* <System>
780 <vssd:Description>Description of the virtual hardware section.</vssd:Description>
781 <vssd:ElementName>vmware</vssd:ElementName>
782 <vssd:InstanceID>1</vssd:InstanceID>
783 <vssd:VirtualSystemIdentifier>MyLampService</vssd:VirtualSystemIdentifier>
784 <vssd:VirtualSystemType>vmx-4</vssd:VirtualSystemType>
785 </System>*/
786 if ((pelmVirtualSystemType = pelmSystem->findChildElement("VirtualSystemType")))
787 vsys.strVirtualSystemType = pelmVirtualSystemType->getValue();
788 }
789
790 xml::NodesLoop loopVirtualHardwareItems(*pelmThis, "Item"); // all "Item" child elements
791 const xml::ElementNode *pelmItem;
792 while ((pelmItem = loopVirtualHardwareItems.forAllNodes()))
793 {
794 VirtualHardwareItem i;
795
796 i.ulLineNumber = pelmItem->getLineNumber();
797
798 xml::NodesLoop loopItemChildren(*pelmItem); // all child elements
799 const xml::ElementNode *pelmItemChild;
800 while ((pelmItemChild = loopItemChildren.forAllNodes()))
801 {
802 const char *pcszItemChildName = pelmItemChild->getName();
803 if (!strcmp(pcszItemChildName, "Description"))
804 i.strDescription = pelmItemChild->getValue();
805 else if (!strcmp(pcszItemChildName, "Caption"))
806 i.strCaption = pelmItemChild->getValue();
807 else if (!strcmp(pcszItemChildName, "ElementName"))
808 i.strElementName = pelmItemChild->getValue();
809 else if ( (!strcmp(pcszItemChildName, "InstanceID"))
810 || (!strcmp(pcszItemChildName, "InstanceId"))
811 )
812 pelmItemChild->copyValue(i.ulInstanceID);
813 else if (!strcmp(pcszItemChildName, "HostResource"))
814 i.strHostResource = pelmItemChild->getValue();
815 else if (!strcmp(pcszItemChildName, "ResourceType"))
816 {
817 uint32_t ulType;
818 pelmItemChild->copyValue(ulType);
819 i.resourceType = (OVFResourceType_T)ulType;
820 }
821 else if (!strcmp(pcszItemChildName, "OtherResourceType"))
822 i.strOtherResourceType = pelmItemChild->getValue();
823 else if (!strcmp(pcszItemChildName, "ResourceSubType"))
824 i.strResourceSubType = pelmItemChild->getValue();
825 else if (!strcmp(pcszItemChildName, "AutomaticAllocation"))
826 i.fAutomaticAllocation = (!strcmp(pelmItemChild->getValue(), "true")) ? true : false;
827 else if (!strcmp(pcszItemChildName, "AutomaticDeallocation"))
828 i.fAutomaticDeallocation = (!strcmp(pelmItemChild->getValue(), "true")) ? true : false;
829 else if (!strcmp(pcszItemChildName, "Parent"))
830 pelmItemChild->copyValue(i.ulParent);
831 else if (!strcmp(pcszItemChildName, "Connection"))
832 i.strConnection = pelmItemChild->getValue();
833 else if (!strcmp(pcszItemChildName, "Address"))
834 i.strAddress = pelmItemChild->getValue();
835 else if (!strcmp(pcszItemChildName, "AddressOnParent"))
836 i.strAddressOnParent = pelmItemChild->getValue();
837 else if (!strcmp(pcszItemChildName, "AllocationUnits"))
838 i.strAllocationUnits = pelmItemChild->getValue();
839 else if (!strcmp(pcszItemChildName, "VirtualQuantity"))
840 pelmItemChild->copyValue(i.ullVirtualQuantity);
841 else if (!strcmp(pcszItemChildName, "Reservation"))
842 pelmItemChild->copyValue(i.ullReservation);
843 else if (!strcmp(pcszItemChildName, "Limit"))
844 pelmItemChild->copyValue(i.ullLimit);
845 else if (!strcmp(pcszItemChildName, "Weight"))
846 pelmItemChild->copyValue(i.ullWeight);
847 else if (!strcmp(pcszItemChildName, "ConsumerVisibility"))
848 i.strConsumerVisibility = pelmItemChild->getValue();
849 else if (!strcmp(pcszItemChildName, "MappingBehavior"))
850 i.strMappingBehavior = pelmItemChild->getValue();
851 else if (!strcmp(pcszItemChildName, "PoolID"))
852 i.strPoolID = pelmItemChild->getValue();
853 else if (!strcmp(pcszItemChildName, "BusNumber"))
854 pelmItemChild->copyValue(i.ulBusNumber);
855 else
856 return setError(VBOX_E_FILE_ERROR,
857 tr("Error reading \"%s\": unknown element \"%s\" under Item element, line %d"),
858 pcszPath,
859 pcszItemChildName,
860 i.ulLineNumber);
861 }
862
863 // store!
864 vsys.mapHardwareItems[i.ulInstanceID] = i;
865 }
866
867 // now go thru all hardware items and handle them according to their type;
868 // in this first loop we handle all items _except_ hard disk images,
869 // which we'll handle in a second loop below
870 HardwareItemsMap::const_iterator itH;
871 for (itH = vsys.mapHardwareItems.begin();
872 itH != vsys.mapHardwareItems.end();
873 ++itH)
874 {
875 const VirtualHardwareItem &i = itH->second;
876
877 // do some analysis
878 switch (i.resourceType)
879 {
880 case OVFResourceType_Processor: // 3
881 /* <rasd:Caption>1 virtual CPU</rasd:Caption>
882 <rasd:Description>Number of virtual CPUs</rasd:Description>
883 <rasd:ElementName>virtual CPU</rasd:ElementName>
884 <rasd:InstanceID>1</rasd:InstanceID>
885 <rasd:ResourceType>3</rasd:ResourceType>
886 <rasd:VirtualQuantity>1</rasd:VirtualQuantity>*/
887 if (i.ullVirtualQuantity < UINT16_MAX)
888 vsys.cCPUs = (uint16_t)i.ullVirtualQuantity;
889 else
890 return setError(VBOX_E_FILE_ERROR,
891 tr("Error reading \"%s\": CPU count %RI64 is larger than %d, line %d"),
892 pcszPath,
893 i.ullVirtualQuantity,
894 UINT16_MAX,
895 i.ulLineNumber);
896 break;
897
898 case OVFResourceType_Memory: // 4
899 if ( (i.strAllocationUnits == "MegaBytes") // found in OVF created by OVF toolkit
900 || (i.strAllocationUnits == "MB") // found in MS docs
901 || (i.strAllocationUnits == "byte * 2^20") // suggested by OVF spec DSP0243 page 21
902 )
903 vsys.ullMemorySize = i.ullVirtualQuantity * 1024 * 1024;
904 else
905 return setError(VBOX_E_FILE_ERROR,
906 tr("Error reading \"%s\": Invalid allocation unit \"%s\" specified with memory size item, line %d"),
907 pcszPath,
908 i.strAllocationUnits.c_str(),
909 i.ulLineNumber);
910 break;
911
912 case OVFResourceType_IDEController: // 5
913 {
914 /* <Item>
915 <rasd:Caption>ideController0</rasd:Caption>
916 <rasd:Description>IDE Controller</rasd:Description>
917 <rasd:InstanceId>5</rasd:InstanceId>
918 <rasd:ResourceType>5</rasd:ResourceType>
919 <rasd:Address>0</rasd:Address>
920 <rasd:BusNumber>0</rasd:BusNumber>
921 </Item> */
922 HardDiskController hdc;
923 hdc.system = HardDiskController::IDE;
924 hdc.idController = i.ulInstanceID;
925 hdc.strControllerType = i.strResourceSubType;
926 hdc.strAddress = i.strAddress;
927 hdc.ulBusNumber = i.ulBusNumber;
928
929 vsys.mapControllers[i.ulInstanceID] = hdc;
930 }
931 break;
932
933 case OVFResourceType_ParallelSCSIHBA: // 6 SCSI controller
934 {
935 /* <Item>
936 <rasd:Caption>SCSI Controller 0 - LSI Logic</rasd:Caption>
937 <rasd:Description>SCI Controller</rasd:Description>
938 <rasd:ElementName>SCSI controller</rasd:ElementName>
939 <rasd:InstanceID>4</rasd:InstanceID>
940 <rasd:ResourceSubType>LsiLogic</rasd:ResourceSubType>
941 <rasd:ResourceType>6</rasd:ResourceType>
942 </Item> */
943 HardDiskController hdc;
944 hdc.system = HardDiskController::SCSI;
945 hdc.idController = i.ulInstanceID;
946 hdc.strControllerType = i.strResourceSubType;
947
948 vsys.mapControllers[i.ulInstanceID] = hdc;
949 }
950 break;
951
952 case OVFResourceType_EthernetAdapter: // 10
953 {
954 /* <Item>
955 <rasd:Caption>Ethernet adapter on 'Bridged'</rasd:Caption>
956 <rasd:AutomaticAllocation>true</rasd:AutomaticAllocation>
957 <rasd:Connection>Bridged</rasd:Connection>
958 <rasd:InstanceID>6</rasd:InstanceID>
959 <rasd:ResourceType>10</rasd:ResourceType>
960 <rasd:ResourceSubType>E1000</rasd:ResourceSubType>
961 </Item>
962
963 OVF spec DSP 0243 page 21:
964 "For an Ethernet adapter, this specifies the abstract network connection name
965 for the virtual machine. All Ethernet adapters that specify the same abstract
966 network connection name within an OVF package shall be deployed on the same
967 network. The abstract network connection name shall be listed in the NetworkSection
968 at the outermost envelope level." */
969
970 // only store the name
971 EthernetAdapter ea;
972 ea.strAdapterType = i.strResourceSubType;
973 ea.strNetworkName = i.strConnection;
974 vsys.llEthernetAdapters.push_back(ea);
975 }
976 break;
977
978 case OVFResourceType_FloppyDrive: // 14
979 vsys.fHasFloppyDrive = true; // we have no additional information
980 break;
981
982 case OVFResourceType_CDDrive: // 15
983 /* <Item ovf:required="false">
984 <rasd:Caption>cdrom1</rasd:Caption>
985 <rasd:InstanceId>7</rasd:InstanceId>
986 <rasd:ResourceType>15</rasd:ResourceType>
987 <rasd:AutomaticAllocation>true</rasd:AutomaticAllocation>
988 <rasd:Parent>5</rasd:Parent>
989 <rasd:AddressOnParent>0</rasd:AddressOnParent>
990 </Item> */
991 // I tried to see what happens if I set an ISO for the CD-ROM in VMware Workstation,
992 // but then the ovftool dies with "Device backing not supported". So I guess if
993 // VMware can't export ISOs, then we don't need to be able to import them right now.
994 vsys.fHasCdromDrive = true; // we have no additional information
995 break;
996
997 case OVFResourceType_HardDisk: // 17
998 // handled separately in second loop below
999 break;
1000
1001 case OVFResourceType_OtherStorageDevice: // 20 SATA controller
1002 {
1003 /* <Item>
1004 <rasd:Description>SATA Controller</rasd:Description>
1005 <rasd:Caption>sataController0</rasd:Caption>
1006 <rasd:InstanceID>4</rasd:InstanceID>
1007 <rasd:ResourceType>20</rasd:ResourceType>
1008 <rasd:ResourceSubType>AHCI</rasd:ResourceSubType>
1009 <rasd:Address>0</rasd:Address>
1010 <rasd:BusNumber>0</rasd:BusNumber>
1011 </Item> */
1012 if (i.strCaption.startsWith ("sataController", Utf8Str::CaseInsensitive) &&
1013 !i.strResourceSubType.compare ("AHCI", Utf8Str::CaseInsensitive))
1014 {
1015 HardDiskController hdc;
1016 hdc.system = HardDiskController::SATA;
1017 hdc.idController = i.ulInstanceID;
1018 hdc.strControllerType = i.strResourceSubType;
1019
1020 vsys.mapControllers[i.ulInstanceID] = hdc;
1021 }
1022 else
1023 return setError(VBOX_E_FILE_ERROR,
1024 tr("Error reading \"%s\": Host resource of type \"Other Storage Device (%d)\" is supported with SATA AHCI controllers only, line %d"),
1025 pcszPath,
1026 OVFResourceType_OtherStorageDevice,
1027 i.ulLineNumber);
1028 }
1029 break;
1030
1031 case OVFResourceType_USBController: // 23
1032 /* <Item ovf:required="false">
1033 <rasd:Caption>usb</rasd:Caption>
1034 <rasd:Description>USB Controller</rasd:Description>
1035 <rasd:InstanceId>3</rasd:InstanceId>
1036 <rasd:ResourceType>23</rasd:ResourceType>
1037 <rasd:Address>0</rasd:Address>
1038 <rasd:BusNumber>0</rasd:BusNumber>
1039 </Item> */
1040 vsys.fHasUsbController = true; // we have no additional information
1041 break;
1042
1043 case OVFResourceType_SoundCard: // 35
1044 /* <Item ovf:required="false">
1045 <rasd:Caption>sound</rasd:Caption>
1046 <rasd:Description>Sound Card</rasd:Description>
1047 <rasd:InstanceId>10</rasd:InstanceId>
1048 <rasd:ResourceType>35</rasd:ResourceType>
1049 <rasd:ResourceSubType>ensoniq1371</rasd:ResourceSubType>
1050 <rasd:AutomaticAllocation>false</rasd:AutomaticAllocation>
1051 <rasd:AddressOnParent>3</rasd:AddressOnParent>
1052 </Item> */
1053 vsys.strSoundCardType = i.strResourceSubType;
1054 break;
1055
1056 default:
1057 return setError(VBOX_E_FILE_ERROR,
1058 tr("Error reading \"%s\": Unknown resource type %d in hardware item, line %d"),
1059 pcszPath,
1060 i.resourceType,
1061 i.ulLineNumber);
1062 } // end switch
1063 }
1064
1065 // now run through the items for a second time, but handle only
1066 // hard disk images; otherwise the code would fail if a hard
1067 // disk image appears in the OVF before its hard disk controller
1068 for (itH = vsys.mapHardwareItems.begin();
1069 itH != vsys.mapHardwareItems.end();
1070 ++itH)
1071 {
1072 const VirtualHardwareItem &i = itH->second;
1073
1074 // do some analysis
1075 switch (i.resourceType)
1076 {
1077 case OVFResourceType_HardDisk: // 17
1078 {
1079 /* <Item>
1080 <rasd:Caption>Harddisk 1</rasd:Caption>
1081 <rasd:Description>HD</rasd:Description>
1082 <rasd:ElementName>Hard Disk</rasd:ElementName>
1083 <rasd:HostResource>ovf://disk/lamp</rasd:HostResource>
1084 <rasd:InstanceID>5</rasd:InstanceID>
1085 <rasd:Parent>4</rasd:Parent>
1086 <rasd:ResourceType>17</rasd:ResourceType>
1087 </Item> */
1088
1089 // look up the hard disk controller element whose InstanceID equals our Parent;
1090 // this is how the connection is specified in OVF
1091 ControllersMap::const_iterator it = vsys.mapControllers.find(i.ulParent);
1092 if (it == vsys.mapControllers.end())
1093 return setError(VBOX_E_FILE_ERROR,
1094 tr("Error reading \"%s\": Hard disk item with instance ID %d specifies invalid parent %d, line %d"),
1095 pcszPath,
1096 i.ulInstanceID,
1097 i.ulParent,
1098 i.ulLineNumber);
1099 //const HardDiskController &hdc = it->second;
1100
1101 VirtualDisk vd;
1102 vd.idController = i.ulParent;
1103 i.strAddressOnParent.toInt(vd.ulAddressOnParent);
1104 // ovf://disk/lamp
1105 // 123456789012345
1106 if (i.strHostResource.substr(0, 11) == "ovf://disk/")
1107 vd.strDiskId = i.strHostResource.substr(11);
1108 else if (i.strHostResource.substr(0, 6) == "/disk/")
1109 vd.strDiskId = i.strHostResource.substr(6);
1110
1111 if ( !(vd.strDiskId.length())
1112 || (m->mapDisks.find(vd.strDiskId) == m->mapDisks.end())
1113 )
1114 return setError(VBOX_E_FILE_ERROR,
1115 tr("Error reading \"%s\": Hard disk item with instance ID %d specifies invalid host resource \"%s\", line %d"),
1116 pcszPath,
1117 i.ulInstanceID,
1118 i.strHostResource.c_str(),
1119 i.ulLineNumber);
1120
1121 vsys.mapVirtualDisks[vd.strDiskId] = vd;
1122 }
1123 break;
1124 }
1125 }
1126 }
1127 else if ( (!strcmp(pcszElemName, "OperatingSystemSection"))
1128 || (!strcmp(pcszTypeAttr, "ovf:OperatingSystemSection_Type"))
1129 )
1130 {
1131 uint64_t cimos64;
1132 if (!(pelmThis->getAttributeValue("id", cimos64)))
1133 return setError(VBOX_E_FILE_ERROR,
1134 tr("Error reading \"%s\": missing or invalid 'ovf:id' attribute in operating system section element, line %d"),
1135 pcszPath,
1136 pelmThis->getLineNumber());
1137
1138 vsys.cimos = (CIMOSType_T)cimos64;
1139 const xml::ElementNode *pelmCIMOSDescription;
1140 if ((pelmCIMOSDescription = pelmThis->findChildElement("Description")))
1141 vsys.strCimosDesc = pelmCIMOSDescription->getValue();
1142 }
1143 else if ( (!strcmp(pcszElemName, "AnnotationSection"))
1144 || (!strcmp(pcszTypeAttr, "ovf:AnnotationSection_Type"))
1145 )
1146 {
1147 const xml::ElementNode *pelmAnnotation;
1148 if ((pelmAnnotation = pelmThis->findChildElement("Annotation")))
1149 vsys.strDescription = pelmAnnotation->getValue();
1150 }
1151 }
1152
1153 // now create the virtual system
1154 m->llVirtualSystems.push_back(vsys);
1155
1156 return S_OK;
1157}
1158
1159////////////////////////////////////////////////////////////////////////////////
1160//
1161// IAppliance public methods
1162//
1163////////////////////////////////////////////////////////////////////////////////
1164
1165/**
1166 * Public method implementation.
1167 * @param
1168 * @return
1169 */
1170STDMETHODIMP Appliance::COMGETTER(Path)(BSTR *aPath)
1171{
1172 if (!aPath)
1173 return E_POINTER;
1174
1175 AutoCaller autoCaller(this);
1176 CheckComRCReturnRC(autoCaller.rc());
1177
1178 AutoReadLock alock(this);
1179
1180 Bstr bstrPath(m->strPath);
1181 bstrPath.cloneTo(aPath);
1182
1183 return S_OK;
1184}
1185
1186/**
1187 * Public method implementation.
1188 * @param
1189 * @return
1190 */
1191STDMETHODIMP Appliance::COMGETTER(Disks)(ComSafeArrayOut(BSTR, aDisks))
1192{
1193 CheckComArgOutSafeArrayPointerValid(aDisks);
1194
1195 AutoCaller autoCaller(this);
1196 CheckComRCReturnRC(autoCaller.rc());
1197
1198 AutoReadLock alock(this);
1199
1200 size_t c = m->mapDisks.size();
1201 com::SafeArray<BSTR> sfaDisks(c);
1202
1203 DiskImagesMap::const_iterator it;
1204 size_t i = 0;
1205 for (it = m->mapDisks.begin();
1206 it != m->mapDisks.end();
1207 ++it, ++i)
1208 {
1209 // create a string representing this disk
1210 const DiskImage &d = it->second;
1211 char *psz = NULL;
1212 RTStrAPrintf(&psz,
1213 "%s\t"
1214 "%RI64\t"
1215 "%RI64\t"
1216 "%s\t"
1217 "%s\t"
1218 "%RI64\t"
1219 "%RI64\t"
1220 "%s",
1221 d.strDiskId.c_str(),
1222 d.iCapacity,
1223 d.iPopulatedSize,
1224 d.strFormat.c_str(),
1225 d.strHref.c_str(),
1226 d.iSize,
1227 d.iChunkSize,
1228 d.strCompression.c_str());
1229 Utf8Str utf(psz);
1230 Bstr bstr(utf);
1231 // push to safearray
1232 bstr.cloneTo(&sfaDisks[i]);
1233 RTStrFree(psz);
1234 }
1235
1236 sfaDisks.detachTo(ComSafeArrayOutArg(aDisks));
1237
1238 return S_OK;
1239}
1240
1241/**
1242 * Public method implementation.
1243 * @param
1244 * @return
1245 */
1246STDMETHODIMP Appliance::COMGETTER(VirtualSystemDescriptions)(ComSafeArrayOut(IVirtualSystemDescription*, aVirtualSystemDescriptions))
1247{
1248 CheckComArgOutSafeArrayPointerValid(aVirtualSystemDescriptions);
1249
1250 AutoCaller autoCaller(this);
1251 CheckComRCReturnRC(autoCaller.rc());
1252
1253 AutoReadLock alock(this);
1254
1255 SafeIfaceArray<IVirtualSystemDescription> sfaVSD(m->virtualSystemDescriptions);
1256 sfaVSD.detachTo(ComSafeArrayOutArg(aVirtualSystemDescriptions));
1257
1258 return S_OK;
1259}
1260
1261/**
1262 * Public method implementation.
1263 * @param path
1264 * @return
1265 */
1266STDMETHODIMP Appliance::Read(IN_BSTR path)
1267{
1268 HRESULT rc = S_OK;
1269
1270 if (!path)
1271 return E_POINTER;
1272
1273 AutoCaller autoCaller(this);
1274 CheckComRCReturnRC(autoCaller.rc());
1275
1276 AutoWriteLock alock(this);
1277
1278 // see if we can handle this file; for now we insist it has an ".ovf" extension
1279 m->strPath = path;
1280 if (!m->strPath.endsWith(".ovf", Utf8Str::CaseInsensitive))
1281 return setError(VBOX_E_FILE_ERROR,
1282 tr("Appliance file must have .ovf extension"));
1283
1284 try
1285 {
1286 xml::XmlFileParser parser;
1287 xml::Document doc;
1288 parser.read(m->strPath.raw(),
1289 doc);
1290
1291 const xml::ElementNode *pRootElem = doc.getRootElement();
1292 if (strcmp(pRootElem->getName(), "Envelope"))
1293 return setError(VBOX_E_FILE_ERROR,
1294 tr("Root element in OVF file must be \"Envelope\"."));
1295
1296 // OVF has the following rough layout:
1297 /*
1298 -- <References> .... files referenced from other parts of the file, such as VMDK images
1299 -- Metadata, comprised of several section commands
1300 -- virtual machines, either a single <VirtualSystem>, or a <VirtualSystemCollection>
1301 -- optionally <Strings> for localization
1302 */
1303
1304 // get all "File" child elements of "References" section so we can look up files easily;
1305 // first find the "References" sections so we can look up files
1306 xml::ElementNodesList listFileElements; // receives all /Envelope/References/File nodes
1307 const xml::ElementNode *pReferencesElem;
1308 if ((pReferencesElem = pRootElem->findChildElement("References")))
1309 pReferencesElem->getChildElements(listFileElements, "File");
1310
1311 // now go though the sections
1312 if (!(SUCCEEDED(rc = LoopThruSections(m->strPath.raw(), pReferencesElem, pRootElem))))
1313 return rc;
1314 }
1315 catch(xml::Error &x)
1316 {
1317 return setError(VBOX_E_FILE_ERROR,
1318 x.what());
1319 }
1320
1321 return S_OK;
1322}
1323
1324/**
1325 * Public method implementation.
1326 * @return
1327 */
1328STDMETHODIMP Appliance::Interpret()
1329{
1330 // @todo:
1331 // - don't use COM methods but the methods directly (faster, but needs appropriate locking of that objects itself (s. HardDisk))
1332 // - Appropriate handle errors like not supported file formats
1333 AutoCaller autoCaller(this);
1334 CheckComRCReturnRC(autoCaller.rc());
1335
1336 AutoWriteLock(this);
1337
1338 HRESULT rc = S_OK;
1339
1340 /* Clear any previous virtual system descriptions */
1341 m->virtualSystemDescriptions.clear();
1342
1343 /* We need the default path for storing disk images */
1344 ComPtr<ISystemProperties> systemProps;
1345 rc = mVirtualBox->COMGETTER(SystemProperties)(systemProps.asOutParam());
1346 CheckComRCReturnRC(rc);
1347 Bstr bstrDefaultHardDiskLocation;
1348 rc = systemProps->COMGETTER(DefaultHardDiskFolder)(bstrDefaultHardDiskLocation.asOutParam());
1349 CheckComRCReturnRC(rc);
1350
1351 /* Try/catch so we can clean up on error */
1352 try
1353 {
1354 list<VirtualSystem>::const_iterator it;
1355 /* Iterate through all virtual systems */
1356 for (it = m->llVirtualSystems.begin();
1357 it != m->llVirtualSystems.end();
1358 ++it)
1359 {
1360 const VirtualSystem &vsysThis = *it;
1361
1362 ComObjPtr<VirtualSystemDescription> pNewDesc;
1363 rc = pNewDesc.createObject();
1364 CheckComRCThrowRC(rc);
1365 rc = pNewDesc->init();
1366 CheckComRCThrowRC(rc);
1367
1368 /* Guest OS type */
1369 Utf8Str strOsTypeVBox,
1370 strCIMOSType = Utf8StrFmt("%RI32", (uint32_t)vsysThis.cimos);
1371 convertCIMOSType2VBoxOSType(strOsTypeVBox, vsysThis.cimos, vsysThis.strCimosDesc);
1372 pNewDesc->addEntry(VirtualSystemDescriptionType_OS,
1373 "",
1374 strCIMOSType,
1375 strOsTypeVBox);
1376
1377 /* VM name */
1378 /* If the there isn't any name specified create a default one out of
1379 * the OS type */
1380 Utf8Str nameVBox = vsysThis.strName;
1381 if (nameVBox.isEmpty())
1382 nameVBox = strOsTypeVBox;
1383 searchUniqueVMName(nameVBox);
1384 pNewDesc->addEntry(VirtualSystemDescriptionType_Name,
1385 "",
1386 vsysThis.strName,
1387 nameVBox);
1388
1389 /* VM Product */
1390 if (!vsysThis.strProduct.isEmpty())
1391 pNewDesc->addEntry(VirtualSystemDescriptionType_Product,
1392 "",
1393 vsysThis.strProduct,
1394 vsysThis.strProduct);
1395
1396 /* VM Vendor */
1397 if (!vsysThis.strVendor.isEmpty())
1398 pNewDesc->addEntry(VirtualSystemDescriptionType_Vendor,
1399 "",
1400 vsysThis.strVendor,
1401 vsysThis.strVendor);
1402
1403 /* VM Version */
1404 if (!vsysThis.strVersion.isEmpty())
1405 pNewDesc->addEntry(VirtualSystemDescriptionType_Version,
1406 "",
1407 vsysThis.strVersion,
1408 vsysThis.strVersion);
1409
1410 /* VM ProductUrl */
1411 if (!vsysThis.strProductUrl.isEmpty())
1412 pNewDesc->addEntry(VirtualSystemDescriptionType_ProductUrl,
1413 "",
1414 vsysThis.strProductUrl,
1415 vsysThis.strProductUrl);
1416
1417 /* VM VendorUrl */
1418 if (!vsysThis.strVendorUrl.isEmpty())
1419 pNewDesc->addEntry(VirtualSystemDescriptionType_VendorUrl,
1420 "",
1421 vsysThis.strVendorUrl,
1422 vsysThis.strVendorUrl);
1423
1424 /* VM description */
1425 if (!vsysThis.strDescription.isEmpty())
1426 pNewDesc->addEntry(VirtualSystemDescriptionType_Description,
1427 "",
1428 vsysThis.strDescription,
1429 vsysThis.strDescription);
1430
1431 /* VM license */
1432 if (!vsysThis.strLicenseText.isEmpty())
1433 pNewDesc->addEntry(VirtualSystemDescriptionType_License,
1434 "",
1435 vsysThis.strLicenseText,
1436 vsysThis.strLicenseText);
1437
1438 /* Now that we know the OS type, get our internal defaults based on that. */
1439 ComPtr<IGuestOSType> pGuestOSType;
1440 rc = mVirtualBox->GetGuestOSType(Bstr(strOsTypeVBox), pGuestOSType.asOutParam());
1441 CheckComRCThrowRC(rc);
1442
1443 /* CPU count */
1444 ULONG cpuCountVBox = vsysThis.cCPUs;
1445 /* Check for the constrains */
1446 if (cpuCountVBox > SchemaDefs::MaxCPUCount)
1447 {
1448 addWarning(tr("The virtual system \"%s\" claims support for %u CPU's, but VirtualBox has support for max %u CPU's only."),
1449 vsysThis.strName.c_str(), cpuCountVBox, SchemaDefs::MaxCPUCount);
1450 cpuCountVBox = SchemaDefs::MaxCPUCount;
1451 }
1452 if (vsysThis.cCPUs == 0)
1453 cpuCountVBox = 1;
1454 pNewDesc->addEntry(VirtualSystemDescriptionType_CPU,
1455 "",
1456 Utf8StrFmt("%RI32", (uint32_t)vsysThis.cCPUs),
1457 Utf8StrFmt("%RI32", (uint32_t)cpuCountVBox));
1458
1459 /* RAM */
1460 uint64_t ullMemSizeVBox = vsysThis.ullMemorySize / _1M;
1461 /* Check for the constrains */
1462 if (ullMemSizeVBox != 0 &&
1463 (ullMemSizeVBox < MM_RAM_MIN_IN_MB ||
1464 ullMemSizeVBox > MM_RAM_MAX_IN_MB))
1465 {
1466 addWarning(tr("The virtual system \"%s\" claims support for %llu MB RAM size, but VirtualBox has support for min %u & max %u MB RAM size only."),
1467 vsysThis.strName.c_str(), ullMemSizeVBox, MM_RAM_MIN_IN_MB, MM_RAM_MAX_IN_MB);
1468 ullMemSizeVBox = RT_MIN(RT_MAX(ullMemSizeVBox, MM_RAM_MIN_IN_MB), MM_RAM_MAX_IN_MB);
1469 }
1470 if (vsysThis.ullMemorySize == 0)
1471 {
1472 /* If the RAM of the OVF is zero, use our predefined values */
1473 ULONG memSizeVBox2;
1474 rc = pGuestOSType->COMGETTER(RecommendedRAM)(&memSizeVBox2);
1475 CheckComRCThrowRC(rc);
1476 /* VBox stores that in MByte */
1477 ullMemSizeVBox = (uint64_t)memSizeVBox2;
1478 }
1479 pNewDesc->addEntry(VirtualSystemDescriptionType_Memory,
1480 "",
1481 Utf8StrFmt("%RI64", (uint64_t)vsysThis.ullMemorySize),
1482 Utf8StrFmt("%RI64", (uint64_t)ullMemSizeVBox));
1483
1484 /* Audio */
1485 if (!vsysThis.strSoundCardType.isNull())
1486 /* Currently we set the AC97 always.
1487 @todo: figure out the hardware which could be possible */
1488 pNewDesc->addEntry(VirtualSystemDescriptionType_SoundCard,
1489 "",
1490 vsysThis.strSoundCardType,
1491 Utf8StrFmt("%RI32", (uint32_t)AudioControllerType_AC97));
1492
1493#ifdef VBOX_WITH_USB
1494 /* USB Controller */
1495 if (vsysThis.fHasUsbController)
1496 pNewDesc->addEntry(VirtualSystemDescriptionType_USBController, "", "", "");
1497#endif /* VBOX_WITH_USB */
1498
1499 /* Network Controller */
1500 size_t cEthernetAdapters = vsysThis.llEthernetAdapters.size();
1501 if (cEthernetAdapters > 0)
1502 {
1503 /* Check for the constrains */
1504 if (cEthernetAdapters > SchemaDefs::NetworkAdapterCount)
1505 addWarning(tr("The virtual system \"%s\" claims support for %zu network adapters, but VirtualBox has support for max %u network adapter only."),
1506 vsysThis.strName.c_str(), cEthernetAdapters, SchemaDefs::NetworkAdapterCount);
1507
1508 /* Get the default network adapter type for the selected guest OS */
1509 NetworkAdapterType_T defaultAdapterVBox = NetworkAdapterType_Am79C970A;
1510 rc = pGuestOSType->COMGETTER(AdapterType)(&defaultAdapterVBox);
1511 CheckComRCThrowRC(rc);
1512
1513 EthernetAdaptersList::const_iterator itEA;
1514 /* Iterate through all abstract networks. We support 8 network
1515 * adapters at the maximum, so the first 8 will be added only. */
1516 size_t a = 0;
1517 for (itEA = vsysThis.llEthernetAdapters.begin();
1518 itEA != vsysThis.llEthernetAdapters.end() && a < SchemaDefs::NetworkAdapterCount;
1519 ++itEA, ++a)
1520 {
1521 const EthernetAdapter &ea = *itEA; // logical network to connect to
1522 Utf8Str strNetwork = ea.strNetworkName;
1523 // make sure it's one of these two
1524 if ( (strNetwork.compare("Null", Utf8Str::CaseInsensitive))
1525 && (strNetwork.compare("NAT", Utf8Str::CaseInsensitive))
1526 && (strNetwork.compare("Bridged", Utf8Str::CaseInsensitive))
1527 && (strNetwork.compare("Internal", Utf8Str::CaseInsensitive))
1528 && (strNetwork.compare("HostOnly", Utf8Str::CaseInsensitive))
1529 )
1530 strNetwork = "Bridged"; // VMware assumes this is the default apparently
1531
1532 /* Figure out the hardware type */
1533 NetworkAdapterType_T nwAdapterVBox = defaultAdapterVBox;
1534 if (!ea.strAdapterType.compare("PCNet32", Utf8Str::CaseInsensitive))
1535 {
1536 /* If the default adapter is already one of the two
1537 * PCNet adapters use the default one. If not use the
1538 * Am79C970A as fallback. */
1539 if (!(defaultAdapterVBox == NetworkAdapterType_Am79C970A ||
1540 defaultAdapterVBox == NetworkAdapterType_Am79C973))
1541 nwAdapterVBox = NetworkAdapterType_Am79C970A;
1542 }
1543#ifdef VBOX_WITH_E1000
1544 /* VMWare accidentally write this with VirtualCenter 3.5,
1545 so make sure in this case always to use the VMWare one */
1546 else if (!ea.strAdapterType.compare("E10000", Utf8Str::CaseInsensitive))
1547 nwAdapterVBox = NetworkAdapterType_I82545EM;
1548 else if (!ea.strAdapterType.compare("E1000", Utf8Str::CaseInsensitive))
1549 {
1550 /* Check if this OVF was written by VirtualBox */
1551 if (vsysThis.strVirtualSystemType.contains("virtualbox", Utf8Str::CaseInsensitive))
1552 {
1553 /* If the default adapter is already one of the three
1554 * E1000 adapters use the default one. If not use the
1555 * I82545EM as fallback. */
1556 if (!(defaultAdapterVBox == NetworkAdapterType_I82540EM ||
1557 defaultAdapterVBox == NetworkAdapterType_I82543GC ||
1558 defaultAdapterVBox == NetworkAdapterType_I82545EM))
1559 nwAdapterVBox = NetworkAdapterType_I82540EM;
1560 }
1561 else
1562 /* Always use this one since it's what VMware uses */
1563 nwAdapterVBox = NetworkAdapterType_I82545EM;
1564 }
1565#endif /* VBOX_WITH_E1000 */
1566
1567 pNewDesc->addEntry(VirtualSystemDescriptionType_NetworkAdapter,
1568 "", // ref
1569 ea.strNetworkName, // orig
1570 Utf8StrFmt("%RI32", (uint32_t)nwAdapterVBox), // conf
1571 0,
1572 Utf8StrFmt("type=%s", strNetwork.c_str())); // extra conf
1573 }
1574 }
1575
1576 /* Floppy Drive */
1577 if (vsysThis.fHasFloppyDrive)
1578 pNewDesc->addEntry(VirtualSystemDescriptionType_Floppy, "", "", "");
1579
1580 /* CD Drive */
1581 /* @todo: I can't disable the CDROM. So nothing to do for now */
1582 /*
1583 if (vsysThis.fHasCdromDrive)
1584 pNewDesc->addEntry(VirtualSystemDescriptionType_CDROM, "", "", "");*/
1585
1586 /* Hard disk Controller */
1587 uint16_t cIDEused = 0;
1588 uint16_t cSATAused = 0; NOREF(cSATAused);
1589 uint16_t cSCSIused = 0; NOREF(cSCSIused);
1590 ControllersMap::const_iterator hdcIt;
1591 /* Iterate through all hard disk controllers */
1592 for (hdcIt = vsysThis.mapControllers.begin();
1593 hdcIt != vsysThis.mapControllers.end();
1594 ++hdcIt)
1595 {
1596 const HardDiskController &hdc = hdcIt->second;
1597 Utf8Str strControllerID = Utf8StrFmt("%RI32", (uint32_t)hdc.idController);
1598
1599 switch (hdc.system)
1600 {
1601 case HardDiskController::IDE:
1602 {
1603 /* Check for the constrains */
1604 /* @todo: I'm very confused! Are these bits *one* controller or
1605 is every port/bus declared as an extra controller. */
1606 if (cIDEused < 4)
1607 {
1608 // @todo: figure out the IDE types
1609 /* Use PIIX4 as default */
1610 Utf8Str strType = "PIIX4";
1611 if (!hdc.strControllerType.compare("PIIX3", Utf8Str::CaseInsensitive))
1612 strType = "PIIX3";
1613 else if (!hdc.strControllerType.compare("ICH6", Utf8Str::CaseInsensitive))
1614 strType = "ICH6";
1615 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerIDE,
1616 strControllerID,
1617 hdc.strControllerType,
1618 strType);
1619 }
1620 else
1621 {
1622 /* Warn only once */
1623 if (cIDEused == 1)
1624 addWarning(tr("The virtual \"%s\" system requests support for more than one IDE controller, but VirtualBox has support for only one."),
1625 vsysThis.strName.c_str());
1626
1627 }
1628 ++cIDEused;
1629 break;
1630 }
1631
1632 case HardDiskController::SATA:
1633 {
1634#ifdef VBOX_WITH_AHCI
1635 /* Check for the constrains */
1636 if (cSATAused < 1)
1637 {
1638 // @todo: figure out the SATA types
1639 /* We only support a plain AHCI controller, so use them always */
1640 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerSATA,
1641 strControllerID,
1642 hdc.strControllerType,
1643 "AHCI");
1644 }
1645 else
1646 {
1647 /* Warn only once */
1648 if (cSATAused == 1)
1649 addWarning(tr("The virtual system \"%s\" requests support for more than one SATA controller, but VirtualBox has support for only one"),
1650 vsysThis.strName.c_str());
1651
1652 }
1653 ++cSATAused;
1654 break;
1655#else /* !VBOX_WITH_AHCI */
1656 addWarning(tr("The virtual system \"%s\" requests at least one SATA controller but this version of VirtualBox does not provide a SATA controller emulation"),
1657 vsysThis.strName.c_str());
1658#endif /* !VBOX_WITH_AHCI */
1659 }
1660
1661 case HardDiskController::SCSI:
1662 {
1663#ifdef VBOX_WITH_LSILOGIC
1664 /* Check for the constrains */
1665 if (cSCSIused < 1)
1666 {
1667 Utf8Str hdcController = "LsiLogic";
1668 if (!hdc.strControllerType.compare("BusLogic", Utf8Str::CaseInsensitive))
1669 hdcController = "BusLogic";
1670 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerSCSI,
1671 strControllerID,
1672 hdc.strControllerType,
1673 hdcController);
1674 }
1675 else
1676 addWarning(tr("The virtual system \"%s\" requests support for an additional SCSI controller of type \"%s\" with ID %s, but VirtualBox presently supports only one SCSI controller."),
1677 vsysThis.strName.c_str(),
1678 hdc.strControllerType.c_str(),
1679 strControllerID.c_str());
1680 ++cSCSIused;
1681 break;
1682#else /* !VBOX_WITH_LSILOGIC */
1683 addWarning(tr("The virtual system \"%s\" requests at least one SATA controller but this version of VirtualBox does not provide a SCSI controller emulation"),
1684 vsysThis.strName.c_str());
1685#endif /* !VBOX_WITH_LSILOGIC */
1686 }
1687 }
1688 }
1689
1690 /* Hard disks */
1691 if (vsysThis.mapVirtualDisks.size() > 0)
1692 {
1693 VirtualDisksMap::const_iterator itVD;
1694 /* Iterate through all hard disks ()*/
1695 for (itVD = vsysThis.mapVirtualDisks.begin();
1696 itVD != vsysThis.mapVirtualDisks.end();
1697 ++itVD)
1698 {
1699 const VirtualDisk &hd = itVD->second;
1700 /* Get the associated disk image */
1701 const DiskImage &di = m->mapDisks[hd.strDiskId];
1702
1703 // @todo:
1704 // - figure out all possible vmdk formats we also support
1705 // - figure out if there is a url specifier for vhd already
1706 // - we need a url specifier for the vdi format
1707 if ( di.strFormat.compare("http://www.vmware.com/specifications/vmdk.html#sparse", Utf8Str::CaseInsensitive)
1708 || di.strFormat.compare("http://www.vmware.com/specifications/vmdk.html#compressed", Utf8Str::CaseInsensitive))
1709 {
1710 /* If the href is empty use the VM name as filename */
1711 Utf8Str strFilename = di.strHref;
1712 if (!strFilename.length())
1713 strFilename = Utf8StrFmt("%s.vmdk", nameVBox.c_str());
1714 /* Construct a unique target path */
1715 Utf8StrFmt strPath("%ls%c%s",
1716 bstrDefaultHardDiskLocation.raw(),
1717 RTPATH_DELIMITER,
1718 strFilename.c_str());
1719 searchUniqueDiskImageFilePath(strPath);
1720
1721 /* find the description for the hard disk controller
1722 * that has the same ID as hd.idController */
1723 const VirtualSystemDescriptionEntry *pController;
1724 if (!(pController = pNewDesc->findControllerFromID(hd.idController)))
1725 throw setError(E_FAIL,
1726 tr("Cannot find hard disk controller with OVF instance ID %RI32 to which disk \"%s\" should be attached"),
1727 hd.idController,
1728 di.strHref.c_str());
1729
1730 /* controller to attach to, and the bus within that controller */
1731 Utf8StrFmt strExtraConfig("controller=%RI16;channel=%RI16",
1732 pController->ulIndex,
1733 hd.ulAddressOnParent);
1734 ULONG ulSize = 0;
1735 if (di.iCapacity != -1)
1736 ulSize = (ULONG)(di.iCapacity / _1M);
1737 else if (di.iPopulatedSize != -1)
1738 ulSize = (ULONG)(di.iPopulatedSize / _1M);
1739 else if (di.iSize != -1)
1740 ulSize = (ULONG)(di.iSize / _1M);
1741 if (ulSize == 0)
1742 ulSize = 10000; // assume 10 GB, this is for the progress bar only anyway
1743 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskImage,
1744 hd.strDiskId,
1745 di.strHref,
1746 strPath,
1747 ulSize,
1748 strExtraConfig);
1749 }
1750 else
1751 throw setError(VBOX_E_FILE_ERROR,
1752 tr("Unsupported format for virtual disk image in OVF: \"%s\"", di.strFormat.c_str()));
1753 }
1754 }
1755
1756 m->virtualSystemDescriptions.push_back(pNewDesc);
1757 }
1758 }
1759 catch (HRESULT aRC)
1760 {
1761 /* On error we clear the list & return */
1762 m->virtualSystemDescriptions.clear();
1763 rc = aRC;
1764 }
1765
1766 return rc;
1767}
1768
1769struct Appliance::TaskImportMachines
1770{
1771 TaskImportMachines(Appliance *aThat, Progress *aProgress)
1772 : pAppliance(aThat)
1773 , progress(aProgress)
1774 , rc(S_OK)
1775 {}
1776 ~TaskImportMachines() {}
1777
1778 HRESULT startThread();
1779
1780 Appliance *pAppliance;
1781 ComObjPtr<Progress> progress;
1782 HRESULT rc;
1783};
1784
1785HRESULT Appliance::TaskImportMachines::startThread()
1786{
1787 int vrc = RTThreadCreate(NULL, Appliance::taskThreadImportMachines, this,
1788 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
1789 "Appliance::Task");
1790 ComAssertMsgRCRet(vrc,
1791 ("Could not create taskThreadImportMachines (%Rrc)\n", vrc), E_FAIL);
1792
1793 return S_OK;
1794}
1795
1796/**
1797 * Public method implementation.
1798 * @param aProgress
1799 * @return
1800 */
1801STDMETHODIMP Appliance::ImportMachines(IProgress **aProgress)
1802{
1803 CheckComArgOutPointerValid(aProgress);
1804
1805 AutoCaller autoCaller(this);
1806 CheckComRCReturnRC(autoCaller.rc());
1807
1808 AutoReadLock(this);
1809
1810 HRESULT rc = S_OK;
1811
1812 ComObjPtr<Progress> progress;
1813 try
1814 {
1815 Bstr progressDesc = BstrFmt(tr("Import appliance '%s'"),
1816 m->strPath.raw());
1817 rc = setUpProgress(progress, progressDesc);
1818 if (FAILED(rc)) throw rc;
1819
1820 /* Initialize our worker task */
1821 std::auto_ptr<TaskImportMachines> task(new TaskImportMachines(this, progress));
1822 //AssertComRCThrowRC (task->autoCaller.rc());
1823
1824 rc = task->startThread();
1825 if (FAILED(rc)) throw rc;
1826
1827 task.release();
1828 }
1829 catch (HRESULT aRC)
1830 {
1831 rc = aRC;
1832 }
1833
1834 if (SUCCEEDED(rc))
1835 /* Return progress to the caller */
1836 progress.queryInterfaceTo(aProgress);
1837
1838 return rc;
1839}
1840
1841struct MyHardDiskAttachment
1842{
1843 Guid uuid;
1844 ComPtr<IMachine> pMachine;
1845 Bstr controllerType;
1846 int32_t lChannel;
1847 int32_t lDevice;
1848};
1849
1850/**
1851 * Worker thread implementation for ImportMachines().
1852 * @param aThread
1853 * @param pvUser
1854 */
1855/* static */
1856DECLCALLBACK(int) Appliance::taskThreadImportMachines(RTTHREAD /* aThread */, void *pvUser)
1857{
1858 std::auto_ptr<TaskImportMachines> task(static_cast<TaskImportMachines*>(pvUser));
1859 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
1860
1861 Appliance *pAppliance = task->pAppliance;
1862
1863 LogFlowFuncEnter();
1864 LogFlowFunc(("Appliance %p\n", pAppliance));
1865
1866 AutoCaller autoCaller(pAppliance);
1867 CheckComRCReturnRC(autoCaller.rc());
1868
1869 AutoWriteLock appLock(pAppliance);
1870
1871 HRESULT rc = S_OK;
1872
1873 ComPtr<IVirtualBox> pVirtualBox(pAppliance->mVirtualBox);
1874
1875 // rollback for errors:
1876 // a list of images that we created/imported
1877 list<MyHardDiskAttachment> llHardDiskAttachments;
1878 list< ComPtr<IHardDisk> > llHardDisksCreated;
1879 list<Guid> llMachinesRegistered;
1880
1881 ComPtr<ISession> session;
1882 bool fSessionOpen = false;
1883 rc = session.createInprocObject(CLSID_Session);
1884 CheckComRCReturnRC(rc);
1885
1886 list<VirtualSystem>::const_iterator it;
1887 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it1;
1888 /* Iterate through all virtual systems of that appliance */
1889 size_t i = 0;
1890 for (it = pAppliance->m->llVirtualSystems.begin(),
1891 it1 = pAppliance->m->virtualSystemDescriptions.begin();
1892 it != pAppliance->m->llVirtualSystems.end();
1893 ++it, ++it1, ++i)
1894 {
1895 const VirtualSystem &vsysThis = *it;
1896 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it1);
1897
1898 ComPtr<IMachine> pNewMachine;
1899
1900 /* Catch possible errors */
1901 try
1902 {
1903 /* Guest OS type */
1904 std::list<VirtualSystemDescriptionEntry*> vsdeOS;
1905 vsdeOS = vsdescThis->findByType(VirtualSystemDescriptionType_OS);
1906 if (vsdeOS.size() < 1)
1907 throw setError(VBOX_E_FILE_ERROR,
1908 tr("Missing guest OS type"));
1909 const Utf8Str &strOsTypeVBox = vsdeOS.front()->strVbox;
1910
1911 /* Now that we know the base system get our internal defaults based on that. */
1912 ComPtr<IGuestOSType> osType;
1913 rc = pVirtualBox->GetGuestOSType(Bstr(strOsTypeVBox), osType.asOutParam());
1914 if (FAILED(rc)) throw rc;
1915
1916 /* Create the machine */
1917 /* First get the name */
1918 std::list<VirtualSystemDescriptionEntry*> vsdeName = vsdescThis->findByType(VirtualSystemDescriptionType_Name);
1919 if (vsdeName.size() < 1)
1920 throw setError(VBOX_E_FILE_ERROR,
1921 tr("Missing VM name"));
1922 const Utf8Str &strNameVBox = vsdeName.front()->strVbox;
1923 rc = pVirtualBox->CreateMachine(Bstr(strNameVBox), Bstr(strOsTypeVBox),
1924 Bstr(), Bstr(),
1925 pNewMachine.asOutParam());
1926 if (FAILED(rc)) throw rc;
1927
1928 // and the description
1929 std::list<VirtualSystemDescriptionEntry*> vsdeDescription = vsdescThis->findByType(VirtualSystemDescriptionType_Description);
1930 if (vsdeDescription.size())
1931 {
1932 const Utf8Str &strDescription = vsdeDescription.front()->strVbox;
1933 rc = pNewMachine->COMSETTER(Description)(Bstr(strDescription));
1934 if (FAILED(rc)) throw rc;
1935 }
1936
1937 /* CPU count */
1938 std::list<VirtualSystemDescriptionEntry*> vsdeCPU = vsdescThis->findByType (VirtualSystemDescriptionType_CPU);
1939 ComAssertMsgThrow(vsdeCPU.size() == 1, ("CPU count missing"), E_FAIL);
1940 const Utf8Str &cpuVBox = vsdeCPU.front()->strVbox;
1941 ULONG tmpCount = (ULONG)RTStrToUInt64(cpuVBox.c_str());
1942 rc = pNewMachine->COMSETTER(CPUCount)(tmpCount);
1943 if (FAILED(rc)) throw rc;
1944 bool fEnableIOApic = false;
1945 /* We need HWVirt & IO-APIC if more than one CPU is requested */
1946 if (tmpCount > 1)
1947 {
1948 rc = pNewMachine->COMSETTER(HWVirtExEnabled)(TSBool_True);
1949 if (FAILED(rc)) throw rc;
1950
1951 fEnableIOApic = true;
1952 }
1953
1954 /* RAM */
1955 std::list<VirtualSystemDescriptionEntry*> vsdeRAM = vsdescThis->findByType(VirtualSystemDescriptionType_Memory);
1956 ComAssertMsgThrow(vsdeRAM.size() == 1, ("RAM size missing"), E_FAIL);
1957 const Utf8Str &memoryVBox = vsdeRAM.front()->strVbox;
1958 ULONG tt = (ULONG)RTStrToUInt64(memoryVBox.c_str());
1959 rc = pNewMachine->COMSETTER(MemorySize)(tt);
1960 if (FAILED(rc)) throw rc;
1961
1962 /* VRAM */
1963 /* Get the recommended VRAM for this guest OS type */
1964 ULONG vramVBox;
1965 rc = osType->COMGETTER(RecommendedVRAM)(&vramVBox);
1966 if (FAILED(rc)) throw rc;
1967
1968 /* Set the VRAM */
1969 rc = pNewMachine->COMSETTER(VRAMSize)(vramVBox);
1970 if (FAILED(rc)) throw rc;
1971
1972 /* I/O APIC: so far we have no setting for this. Enable it if we
1973 import a Windows VM because if if Windows was installed without IOAPIC,
1974 it will not mind finding an one later on, but if Windows was installed
1975 _with_ an IOAPIC, it will bluescreen if it's not found */
1976 Bstr bstrFamilyId;
1977 rc = osType->COMGETTER(FamilyId)(bstrFamilyId.asOutParam());
1978 if (FAILED(rc)) throw rc;
1979
1980 Utf8Str strFamilyId(bstrFamilyId);
1981 if (strFamilyId == "Windows")
1982 fEnableIOApic = true;
1983
1984 /* If IP-APIC should be enabled could be have different reasons.
1985 See CPU count & the Win test above. Here we enable it if it was
1986 previously requested. */
1987 if (fEnableIOApic)
1988 {
1989 ComPtr<IBIOSSettings> pBIOSSettings;
1990 rc = pNewMachine->COMGETTER(BIOSSettings)(pBIOSSettings.asOutParam());
1991 if (FAILED(rc)) throw rc;
1992
1993 rc = pBIOSSettings->COMSETTER(IOAPICEnabled)(TRUE);
1994 if (FAILED(rc)) throw rc;
1995 }
1996
1997 /* Audio Adapter */
1998 std::list<VirtualSystemDescriptionEntry*> vsdeAudioAdapter = vsdescThis->findByType(VirtualSystemDescriptionType_SoundCard);
1999 /* @todo: we support one audio adapter only */
2000 if (vsdeAudioAdapter.size() > 0)
2001 {
2002 const Utf8Str& audioAdapterVBox = vsdeAudioAdapter.front()->strVbox;
2003 if (audioAdapterVBox.compare("null", Utf8Str::CaseInsensitive) != 0)
2004 {
2005 uint32_t audio = RTStrToUInt32(audioAdapterVBox.c_str());
2006 ComPtr<IAudioAdapter> audioAdapter;
2007 rc = pNewMachine->COMGETTER(AudioAdapter)(audioAdapter.asOutParam());
2008 if (FAILED(rc)) throw rc;
2009 rc = audioAdapter->COMSETTER(Enabled)(true);
2010 if (FAILED(rc)) throw rc;
2011 rc = audioAdapter->COMSETTER(AudioController)(static_cast<AudioControllerType_T>(audio));
2012 if (FAILED(rc)) throw rc;
2013 }
2014 }
2015
2016#ifdef VBOX_WITH_USB
2017 /* USB Controller */
2018 std::list<VirtualSystemDescriptionEntry*> vsdeUSBController = vsdescThis->findByType(VirtualSystemDescriptionType_USBController);
2019 // USB support is enabled if there's at least one such entry; to disable USB support,
2020 // the type of the USB item would have been changed to "ignore"
2021 bool fUSBEnabled = vsdeUSBController.size() > 0;
2022
2023 ComPtr<IUSBController> usbController;
2024 rc = pNewMachine->COMGETTER(USBController)(usbController.asOutParam());
2025 if (FAILED(rc)) throw rc;
2026 rc = usbController->COMSETTER(Enabled)(fUSBEnabled);
2027 if (FAILED(rc)) throw rc;
2028#endif /* VBOX_WITH_USB */
2029
2030 /* Change the network adapters */
2031 std::list<VirtualSystemDescriptionEntry*> vsdeNW = vsdescThis->findByType(VirtualSystemDescriptionType_NetworkAdapter);
2032 if (vsdeNW.size() == 0)
2033 {
2034 /* No network adapters, so we have to disable our default one */
2035 ComPtr<INetworkAdapter> nwVBox;
2036 rc = pNewMachine->GetNetworkAdapter(0, nwVBox.asOutParam());
2037 if (FAILED(rc)) throw rc;
2038 rc = nwVBox->COMSETTER(Enabled)(false);
2039 if (FAILED(rc)) throw rc;
2040 }
2041 else
2042 {
2043 list<VirtualSystemDescriptionEntry*>::const_iterator nwIt;
2044 /* Iterate through all network cards. We support 8 network adapters
2045 * at the maximum. (@todo: warn if there are more!) */
2046 size_t a = 0;
2047 for (nwIt = vsdeNW.begin();
2048 (nwIt != vsdeNW.end() && a < SchemaDefs::NetworkAdapterCount);
2049 ++nwIt, ++a)
2050 {
2051 const VirtualSystemDescriptionEntry* pvsys = *nwIt;
2052
2053 const Utf8Str &nwTypeVBox = pvsys->strVbox;
2054 uint32_t tt1 = RTStrToUInt32(nwTypeVBox.c_str());
2055 ComPtr<INetworkAdapter> pNetworkAdapter;
2056 rc = pNewMachine->GetNetworkAdapter((ULONG)a, pNetworkAdapter.asOutParam());
2057 if (FAILED(rc)) throw rc;
2058 /* Enable the network card & set the adapter type */
2059 rc = pNetworkAdapter->COMSETTER(Enabled)(true);
2060 if (FAILED(rc)) throw rc;
2061 rc = pNetworkAdapter->COMSETTER(AdapterType)(static_cast<NetworkAdapterType_T>(tt1));
2062 if (FAILED(rc)) throw rc;
2063
2064 // default is NAT; change to "bridged" if extra conf says so
2065 if (!pvsys->strExtraConfig.compare("type=Bridged", Utf8Str::CaseInsensitive))
2066 {
2067 /* Attach to the right interface */
2068 rc = pNetworkAdapter->AttachToBridgedInterface();
2069 if (FAILED(rc)) throw rc;
2070 ComPtr<IHost> host;
2071 rc = pVirtualBox->COMGETTER(Host)(host.asOutParam());
2072 if (FAILED(rc)) throw rc;
2073 com::SafeIfaceArray<IHostNetworkInterface> nwInterfaces;
2074 rc = host->COMGETTER(NetworkInterfaces)(ComSafeArrayAsOutParam(nwInterfaces));
2075 if (FAILED(rc)) throw rc;
2076 /* We search for the first host network interface which
2077 * is usable for bridged networking */
2078 for (size_t i=0; i < nwInterfaces.size(); ++i)
2079 {
2080 HostNetworkInterfaceType_T itype;
2081 rc = nwInterfaces[i]->COMGETTER(InterfaceType)(&itype);
2082 if (FAILED(rc)) throw rc;
2083 if (itype == HostNetworkInterfaceType_Bridged)
2084 {
2085 Bstr name;
2086 rc = nwInterfaces[i]->COMGETTER(Name)(name.asOutParam());
2087 if (FAILED(rc)) throw rc;
2088 /* Set the interface name to attach to */
2089 pNetworkAdapter->COMSETTER(HostInterface)(name);
2090 if (FAILED(rc)) throw rc;
2091 break;
2092 }
2093 }
2094 }
2095 /* Next test for host only interfaces */
2096 else if (!pvsys->strExtraConfig.compare("type=HostOnly", Utf8Str::CaseInsensitive))
2097 {
2098 /* Attach to the right interface */
2099 rc = pNetworkAdapter->AttachToHostOnlyInterface();
2100 if (FAILED(rc)) throw rc;
2101 ComPtr<IHost> host;
2102 rc = pVirtualBox->COMGETTER(Host)(host.asOutParam());
2103 if (FAILED(rc)) throw rc;
2104 com::SafeIfaceArray<IHostNetworkInterface> nwInterfaces;
2105 rc = host->COMGETTER(NetworkInterfaces)(ComSafeArrayAsOutParam(nwInterfaces));
2106 if (FAILED(rc)) throw rc;
2107 /* We search for the first host network interface which
2108 * is usable for host only networking */
2109 for (size_t i=0; i < nwInterfaces.size(); ++i)
2110 {
2111 HostNetworkInterfaceType_T itype;
2112 rc = nwInterfaces[i]->COMGETTER(InterfaceType)(&itype);
2113 if (FAILED(rc)) throw rc;
2114 if (itype == HostNetworkInterfaceType_HostOnly)
2115 {
2116 Bstr name;
2117 rc = nwInterfaces[i]->COMGETTER(Name)(name.asOutParam());
2118 if (FAILED(rc)) throw rc;
2119 /* Set the interface name to attach to */
2120 pNetworkAdapter->COMSETTER(HostInterface)(name);
2121 if (FAILED(rc)) throw rc;
2122 break;
2123 }
2124 }
2125 }
2126 }
2127 }
2128
2129 /* Floppy drive */
2130 std::list<VirtualSystemDescriptionEntry*> vsdeFloppy = vsdescThis->findByType(VirtualSystemDescriptionType_Floppy);
2131 // Floppy support is enabled if there's at least one such entry; to disable floppy support,
2132 // the type of the floppy item would have been changed to "ignore"
2133 bool fFloppyEnabled = vsdeFloppy.size() > 0;
2134 ComPtr<IFloppyDrive> floppyDrive;
2135 rc = pNewMachine->COMGETTER(FloppyDrive)(floppyDrive.asOutParam());
2136 if (FAILED(rc)) throw rc;
2137 rc = floppyDrive->COMSETTER(Enabled)(fFloppyEnabled);
2138 if (FAILED(rc)) throw rc;
2139
2140 /* CDROM drive */
2141 /* @todo: I can't disable the CDROM. So nothing to do for now */
2142 // std::list<VirtualSystemDescriptionEntry*> vsdeFloppy = vsd->findByType(VirtualSystemDescriptionType_CDROM);
2143
2144 /* Hard disk controller IDE */
2145 std::list<VirtualSystemDescriptionEntry*> vsdeHDCIDE = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskControllerIDE);
2146 if (vsdeHDCIDE.size() > 1)
2147 throw setError(VBOX_E_FILE_ERROR,
2148 tr("Too many IDE controllers in OVF; VirtualBox only supports one"));
2149 if (vsdeHDCIDE.size() == 1)
2150 {
2151 ComPtr<IStorageController> pController;
2152 rc = pNewMachine->GetStorageControllerByName(Bstr("IDE"), pController.asOutParam());
2153 if (FAILED(rc)) throw rc;
2154
2155 const char *pcszIDEType = vsdeHDCIDE.front()->strVbox.c_str();
2156 if (!strcmp(pcszIDEType, "PIIX3"))
2157 rc = pController->COMSETTER(ControllerType)(StorageControllerType_PIIX3);
2158 else if (!strcmp(pcszIDEType, "PIIX4"))
2159 rc = pController->COMSETTER(ControllerType)(StorageControllerType_PIIX4);
2160 else if (!strcmp(pcszIDEType, "ICH6"))
2161 rc = pController->COMSETTER(ControllerType)(StorageControllerType_ICH6);
2162 else
2163 throw setError(VBOX_E_FILE_ERROR,
2164 tr("Invalid IDE controller type \"%s\""),
2165 pcszIDEType);
2166 if (FAILED(rc)) throw rc;
2167 }
2168#ifdef VBOX_WITH_AHCI
2169 /* Hard disk controller SATA */
2170 std::list<VirtualSystemDescriptionEntry*> vsdeHDCSATA = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskControllerSATA);
2171 if (vsdeHDCSATA.size() > 1)
2172 throw setError(VBOX_E_FILE_ERROR,
2173 tr("Too many SATA controllers in OVF; VirtualBox only supports one"));
2174 if (vsdeHDCSATA.size() > 0)
2175 {
2176 ComPtr<IStorageController> pController;
2177 const Utf8Str &hdcVBox = vsdeHDCSATA.front()->strVbox;
2178 if (hdcVBox == "AHCI")
2179 {
2180 rc = pNewMachine->AddStorageController(Bstr("SATA"), StorageBus_SATA, pController.asOutParam());
2181 if (FAILED(rc)) throw rc;
2182 }
2183 else
2184 throw setError(VBOX_E_FILE_ERROR,
2185 tr("Invalid SATA controller type \"%s\""),
2186 hdcVBox.c_str());
2187 }
2188#endif /* VBOX_WITH_AHCI */
2189
2190#ifdef VBOX_WITH_LSILOGIC
2191 /* Hard disk controller SCSI */
2192 std::list<VirtualSystemDescriptionEntry*> vsdeHDCSCSI = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskControllerSCSI);
2193 if (vsdeHDCSCSI.size() > 1)
2194 throw setError(VBOX_E_FILE_ERROR,
2195 tr("Too many SCSI controllers in OVF; VirtualBox only supports one"));
2196 if (vsdeHDCSCSI.size() > 0)
2197 {
2198 ComPtr<IStorageController> pController;
2199 StorageControllerType_T controllerType;
2200 const Utf8Str &hdcVBox = vsdeHDCSCSI.front()->strVbox;
2201 if (hdcVBox == "LsiLogic")
2202 controllerType = StorageControllerType_LsiLogic;
2203 else if (hdcVBox == "BusLogic")
2204 controllerType = StorageControllerType_BusLogic;
2205 else
2206 throw setError(VBOX_E_FILE_ERROR,
2207 tr("Invalid SCSI controller type \"%s\""),
2208 hdcVBox.c_str());
2209
2210 rc = pNewMachine->AddStorageController(Bstr("SCSI"), StorageBus_SCSI, pController.asOutParam());
2211 if (FAILED(rc)) throw rc;
2212 rc = pController->COMSETTER(ControllerType)(controllerType);
2213 if (FAILED(rc)) throw rc;
2214 }
2215#endif /* VBOX_WITH_LSILOGIC */
2216
2217 /* Now its time to register the machine before we add any hard disks */
2218 rc = pVirtualBox->RegisterMachine(pNewMachine);
2219 if (FAILED(rc)) throw rc;
2220
2221 Bstr newMachineId_;
2222 rc = pNewMachine->COMGETTER(Id)(newMachineId_.asOutParam());
2223 if (FAILED(rc)) throw rc;
2224 Guid newMachineId(newMachineId_);
2225
2226 // store new machine for roll-back in case of errors
2227 llMachinesRegistered.push_back(newMachineId);
2228
2229 /* Create the hard disks & connect them to the appropriate controllers. */
2230 std::list<VirtualSystemDescriptionEntry*> avsdeHDs = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskImage);
2231 if (avsdeHDs.size() > 0)
2232 {
2233 /* If in the next block an error occur we have to deregister
2234 the machine, so make an extra try/catch block. */
2235 ComPtr<IHardDisk> srcHdVBox;
2236 bool fSourceHdNeedsClosing = false;
2237
2238 try
2239 {
2240 /* In order to attach hard disks we need to open a session
2241 * for the new machine */
2242 rc = pVirtualBox->OpenSession(session, newMachineId_);
2243 if (FAILED(rc)) throw rc;
2244 fSessionOpen = true;
2245
2246 /* The disk image has to be on the same place as the OVF file. So
2247 * strip the filename out of the full file path. */
2248 Utf8Str strSrcDir = stripFilename(pAppliance->m->strPath);
2249
2250 /* Iterate over all given disk images */
2251 list<VirtualSystemDescriptionEntry*>::const_iterator itHD;
2252 for (itHD = avsdeHDs.begin();
2253 itHD != avsdeHDs.end();
2254 ++itHD)
2255 {
2256 VirtualSystemDescriptionEntry *vsdeHD = *itHD;
2257
2258 const char *pcszDstFilePath = vsdeHD->strVbox.c_str();
2259 /* Check if the destination file exists already or the
2260 * destination path is empty. */
2261 if ( !(*pcszDstFilePath)
2262 || RTPathExists(pcszDstFilePath)
2263 )
2264 /* This isn't allowed */
2265 throw setError(VBOX_E_FILE_ERROR,
2266 tr("Destination file '%s' exists",
2267 pcszDstFilePath));
2268
2269 /* Find the disk from the OVF's disk list */
2270 DiskImagesMap::const_iterator itDiskImage = pAppliance->m->mapDisks.find(vsdeHD->strRef);
2271 /* vsdeHD->strRef contains the disk identifier (e.g. "vmdisk1"), which should exist
2272 in the virtual system's disks map under that ID and also in the global images map. */
2273 VirtualDisksMap::const_iterator itVirtualDisk = vsysThis.mapVirtualDisks.find(vsdeHD->strRef);
2274
2275 if ( itDiskImage == pAppliance->m->mapDisks.end()
2276 || itVirtualDisk == vsysThis.mapVirtualDisks.end()
2277 )
2278 throw setError(E_FAIL,
2279 tr("Internal inconsistency looking up disk images."));
2280
2281 const DiskImage &di = itDiskImage->second;
2282 const VirtualDisk &vd = itVirtualDisk->second;
2283
2284 /* Make sure all target directories exists */
2285 rc = VirtualBox::ensureFilePathExists(pcszDstFilePath);
2286 if (FAILED(rc))
2287 throw rc;
2288
2289 // subprogress object for hard disk
2290 ComPtr<IProgress> pProgress2;
2291
2292 ComPtr<IHardDisk> dstHdVBox;
2293 /* If strHref is empty we have to create a new file */
2294 if (di.strHref.isEmpty())
2295 {
2296 /* Which format to use? */
2297 Bstr srcFormat = L"VDI";
2298 if ( di.strFormat.compare("http://www.vmware.com/specifications/vmdk.html#sparse", Utf8Str::CaseInsensitive)
2299 || di.strFormat.compare("http://www.vmware.com/specifications/vmdk.html#compressed", Utf8Str::CaseInsensitive))
2300 srcFormat = L"VMDK";
2301 /* Create an empty hard disk */
2302 rc = pVirtualBox->CreateHardDisk(srcFormat, Bstr(pcszDstFilePath), dstHdVBox.asOutParam());
2303 if (FAILED(rc)) throw rc;
2304
2305 /* Create a dynamic growing disk image with the given capacity */
2306 rc = dstHdVBox->CreateBaseStorage(di.iCapacity / _1M, HardDiskVariant_Standard, pProgress2.asOutParam());
2307 if (FAILED(rc)) throw rc;
2308
2309 /* Advance to the next operation */
2310 if (!task->progress.isNull())
2311 task->progress->setNextOperation(BstrFmt(tr("Creating virtual disk image '%s'"), pcszDstFilePath),
2312 vsdeHD->ulSizeMB); // operation's weight, as set up with the IProgress originally
2313 }
2314 else
2315 {
2316 /* Construct the source file path */
2317 Utf8StrFmt strSrcFilePath("%s%c%s", strSrcDir.c_str(), RTPATH_DELIMITER, di.strHref.c_str());
2318 /* Check if the source file exists */
2319 if (!RTPathExists(strSrcFilePath.c_str()))
2320 /* This isn't allowed */
2321 throw setError(VBOX_E_FILE_ERROR,
2322 tr("Source virtual disk image file '%s' doesn't exist"),
2323 strSrcFilePath.c_str());
2324
2325 /* Clone the disk image (this is necessary cause the id has
2326 * to be recreated for the case the same hard disk is
2327 * attached already from a previous import) */
2328
2329 /* First open the existing disk image */
2330 rc = pVirtualBox->OpenHardDisk(Bstr(strSrcFilePath),
2331 AccessMode_ReadOnly,
2332 false, Bstr(""), false, Bstr(""),
2333 srcHdVBox.asOutParam());
2334 if (FAILED(rc)) throw rc;
2335 fSourceHdNeedsClosing = true;
2336
2337 /* We need the format description of the source disk image */
2338 Bstr srcFormat;
2339 rc = srcHdVBox->COMGETTER(Format)(srcFormat.asOutParam());
2340 if (FAILED(rc)) throw rc;
2341 /* Create a new hard disk interface for the destination disk image */
2342 rc = pVirtualBox->CreateHardDisk(srcFormat, Bstr(pcszDstFilePath), dstHdVBox.asOutParam());
2343 if (FAILED(rc)) throw rc;
2344 /* Clone the source disk image */
2345 rc = srcHdVBox->CloneTo(dstHdVBox, HardDiskVariant_Standard, NULL, pProgress2.asOutParam());
2346 if (FAILED(rc)) throw rc;
2347
2348 /* Advance to the next operation */
2349 if (!task->progress.isNull())
2350 task->progress->setNextOperation(BstrFmt(tr("Importing virtual disk image '%s'"), strSrcFilePath.c_str()),
2351 vsdeHD->ulSizeMB); // operation's weight, as set up with the IProgress originally);
2352 }
2353
2354 // now wait for the background disk operation to complete; this throws HRESULTs on error
2355 pAppliance->waitForAsyncProgress(task->progress, pProgress2);
2356
2357 if (fSourceHdNeedsClosing)
2358 {
2359 rc = srcHdVBox->Close();
2360 if (FAILED(rc)) throw rc;
2361 fSourceHdNeedsClosing = false;
2362 }
2363
2364 llHardDisksCreated.push_back(dstHdVBox);
2365 /* Now use the new uuid to attach the disk image to our new machine */
2366 ComPtr<IMachine> sMachine;
2367 rc = session->COMGETTER(Machine)(sMachine.asOutParam());
2368 if (FAILED(rc)) throw rc;
2369 Bstr hdId;
2370 rc = dstHdVBox->COMGETTER(Id)(hdId.asOutParam());
2371 if (FAILED(rc)) throw rc;
2372
2373 /* For now we assume we have one controller of every type only */
2374 HardDiskController hdc = (*vsysThis.mapControllers.find(vd.idController)).second;
2375
2376 // this is for rollback later
2377 MyHardDiskAttachment mhda;
2378 mhda.uuid = newMachineId;
2379 mhda.pMachine = pNewMachine;
2380
2381 switch (hdc.system)
2382 {
2383 case HardDiskController::IDE:
2384 // For the IDE bus, the channel parameter can be either 0 or 1, to specify the primary
2385 // or secondary IDE controller, respectively. For the primary controller of the IDE bus,
2386 // the device number can be either 0 or 1, to specify the master or the slave device,
2387 // respectively. For the secondary IDE controller, the device number is always 1 because
2388 // the master device is reserved for the CD-ROM drive.
2389 mhda.controllerType = Bstr("IDE");
2390 switch (vd.ulAddressOnParent)
2391 {
2392 case 0: // interpret this as primary master
2393 mhda.lChannel = (long)0;
2394 mhda.lDevice = (long)0;
2395 break;
2396
2397 case 1: // interpret this as primary slave
2398 mhda.lChannel = (long)0;
2399 mhda.lDevice = (long)1;
2400 break;
2401
2402 case 2: // interpret this as secondary slave
2403 mhda.lChannel = (long)1;
2404 mhda.lDevice = (long)1;
2405 break;
2406
2407 default:
2408 throw setError(VBOX_E_NOT_SUPPORTED,
2409 tr("Invalid channel %RI16 specified; IDE controllers support only 0, 1 or 2"), vd.ulAddressOnParent);
2410 break;
2411 }
2412 break;
2413
2414 case HardDiskController::SATA:
2415 mhda.controllerType = Bstr("SATA");
2416 mhda.lChannel = (long)vd.ulAddressOnParent;
2417 mhda.lDevice = (long)0;
2418 break;
2419
2420 case HardDiskController::SCSI:
2421 mhda.controllerType = Bstr("SCSI");
2422 mhda.lChannel = (long)vd.ulAddressOnParent;
2423 mhda.lDevice = (long)0;
2424 break;
2425
2426 default: break;
2427 }
2428
2429 Log(("Attaching disk %s to channel %d on device %d\n", pcszDstFilePath, mhda.lChannel, mhda.lDevice));
2430
2431 rc = sMachine->AttachHardDisk(hdId,
2432 mhda.controllerType,
2433 mhda.lChannel,
2434 mhda.lDevice);
2435 if (FAILED(rc)) throw rc;
2436
2437 llHardDiskAttachments.push_back(mhda);
2438
2439 rc = sMachine->SaveSettings();
2440 if (FAILED(rc)) throw rc;
2441 } // end for (itHD = avsdeHDs.begin();
2442
2443 // only now that we're done with all disks, close the session
2444 rc = session->Close();
2445 if (FAILED(rc)) throw rc;
2446 fSessionOpen = false;
2447 }
2448 catch(HRESULT /* aRC */)
2449 {
2450 if (fSourceHdNeedsClosing)
2451 srcHdVBox->Close();
2452
2453 if (fSessionOpen)
2454 session->Close();
2455
2456 throw;
2457 }
2458 }
2459 }
2460 catch(HRESULT aRC)
2461 {
2462 rc = aRC;
2463 }
2464
2465 if (FAILED(rc))
2466 break;
2467
2468 } // for (it = pAppliance->m->llVirtualSystems.begin(),
2469
2470 if (FAILED(rc))
2471 {
2472 // with _whatever_ error we've had, do a complete roll-back of
2473 // machines and disks we've created; unfortunately this is
2474 // not so trivially done...
2475
2476 HRESULT rc2;
2477 // detach all hard disks from all machines we created
2478 list<MyHardDiskAttachment>::iterator itM;
2479 for (itM = llHardDiskAttachments.begin();
2480 itM != llHardDiskAttachments.end();
2481 ++itM)
2482 {
2483 const MyHardDiskAttachment &mhda = *itM;
2484 rc2 = pVirtualBox->OpenSession(session, Bstr(mhda.uuid));
2485 if (SUCCEEDED(rc2))
2486 {
2487 ComPtr<IMachine> sMachine;
2488 rc2 = session->COMGETTER(Machine)(sMachine.asOutParam());
2489 if (SUCCEEDED(rc2))
2490 {
2491 rc2 = sMachine->DetachHardDisk(Bstr(mhda.controllerType), mhda.lChannel, mhda.lDevice);
2492 rc2 = sMachine->SaveSettings();
2493 }
2494 session->Close();
2495 }
2496 }
2497
2498 // now clean up all hard disks we created
2499 list< ComPtr<IHardDisk> >::iterator itHD;
2500 for (itHD = llHardDisksCreated.begin();
2501 itHD != llHardDisksCreated.end();
2502 ++itHD)
2503 {
2504 ComPtr<IHardDisk> pDisk = *itHD;
2505 ComPtr<IProgress> pProgress;
2506 rc2 = pDisk->DeleteStorage(pProgress.asOutParam());
2507 rc2 = pProgress->WaitForCompletion(-1);
2508 }
2509
2510 // finally, deregister and remove all machines
2511 list<Guid>::iterator itID;
2512 for (itID = llMachinesRegistered.begin();
2513 itID != llMachinesRegistered.end();
2514 ++itID)
2515 {
2516 const Guid &guid = *itID;
2517 ComPtr<IMachine> failedMachine;
2518 rc2 = pVirtualBox->UnregisterMachine(guid.toUtf16(), failedMachine.asOutParam());
2519 if (SUCCEEDED(rc2))
2520 rc2 = failedMachine->DeleteSettings();
2521 }
2522 }
2523
2524 task->rc = rc;
2525
2526 if (!task->progress.isNull())
2527 task->progress->notifyComplete(rc);
2528
2529 LogFlowFunc(("rc=%Rhrc\n", rc));
2530 LogFlowFuncLeave();
2531
2532 return VINF_SUCCESS;
2533}
2534
2535struct Appliance::TaskWriteOVF
2536{
2537 enum OVFFormat
2538 {
2539 unspecified,
2540 OVF_0_9,
2541 OVF_1_0
2542 };
2543 enum TaskType
2544 {
2545 Write
2546 };
2547
2548 TaskWriteOVF(OVFFormat aFormat, Appliance *aThat)
2549 : taskType(Write),
2550 storageType(VFSType_File),
2551 enFormat(aFormat),
2552 pAppliance(aThat),
2553 rc(S_OK)
2554 {}
2555 ~TaskWriteOVF() {}
2556
2557 int startThread();
2558 static int uploadProgress(unsigned uPercent, void *pvUser);
2559
2560 TaskType taskType;
2561 VFSType_T storageType;
2562 Utf8Str filepath;
2563 Utf8Str hostname;
2564 Utf8Str username;
2565 Utf8Str password;
2566 OVFFormat enFormat;
2567 Appliance *pAppliance;
2568 ComObjPtr<Progress> progress;
2569 HRESULT rc;
2570};
2571
2572int Appliance::TaskWriteOVF::startThread()
2573{
2574 int vrc = RTThreadCreate(NULL, Appliance::taskThreadWriteOVF, this,
2575 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
2576 "Appliance::Task");
2577
2578 ComAssertMsgRCRet(vrc,
2579 ("Could not create taskThreadWriteOVF (%Rrc)\n", vrc), E_FAIL);
2580
2581 return S_OK;
2582}
2583
2584/* static */
2585int Appliance::TaskWriteOVF::uploadProgress(unsigned uPercent, void *pvUser)
2586{
2587 Appliance::TaskWriteOVF* pTask = *(Appliance::TaskWriteOVF**)pvUser;
2588
2589 if (pTask &&
2590 !pTask->progress.isNull())
2591 {
2592 BOOL fCanceled;
2593 pTask->progress->COMGETTER(Canceled)(&fCanceled);
2594 if (fCanceled)
2595 return -1;
2596 pTask->progress->setCurrentOperationProgress(uPercent);
2597 }
2598 return VINF_SUCCESS;
2599}
2600
2601STDMETHODIMP Appliance::CreateVFSExplorer(IN_BSTR aURI, IVFSExplorer **aExplorer)
2602{
2603 HRESULT rc = S_OK;
2604
2605 CheckComArgOutPointerValid(aExplorer);
2606
2607 AutoCaller autoCaller(this);
2608 CheckComRCReturnRC(autoCaller.rc());
2609
2610 AutoReadLock(this);
2611
2612 Utf8Str uri(aURI);
2613 /* Check which kind of export the user has requested */
2614 VFSType_T type = VFSType_File;
2615 Utf8Str strProtocol = "";
2616 /* Check the URI for the target format */
2617 if (uri.startsWith("SunCloud://", Utf8Str::CaseInsensitive)) /* Sun Cloud service */
2618 {
2619 throw E_NOTIMPL;
2620// type = VFSType_S3;
2621// strProtocol = "SunCloud://";
2622 }
2623 else if (uri.startsWith("S3://", Utf8Str::CaseInsensitive)) /* S3 service */
2624 {
2625 throw E_NOTIMPL;
2626// type = VFSType_S3;
2627// strProtocol = "S3://";
2628 }
2629 else if (uri.startsWith("webdav://", Utf8Str::CaseInsensitive)) /* webdav service */
2630 throw E_NOTIMPL;
2631
2632 Utf8Str strFilepath;
2633 Utf8Str strHostname;
2634 Utf8Str strUsername;
2635 Utf8Str strPassword;
2636 parseURI(uri, strProtocol, strFilepath, strHostname, strUsername, strPassword);
2637
2638 ComObjPtr<VFSExplorer> explorer;
2639 explorer.createObject();
2640
2641 rc = explorer->init(type, strFilepath, strHostname, strUsername, strPassword, mVirtualBox);
2642
2643 if (SUCCEEDED(rc))
2644 /* Return explorer to the caller */
2645 explorer.queryInterfaceTo(aExplorer);
2646
2647 return rc;
2648}
2649
2650STDMETHODIMP Appliance::Write(IN_BSTR format, IN_BSTR path, IProgress **aProgress)
2651{
2652 HRESULT rc = S_OK;
2653
2654 CheckComArgOutPointerValid(aProgress);
2655
2656 AutoCaller autoCaller(this);
2657 CheckComRCReturnRC(autoCaller.rc());
2658
2659 AutoWriteLock(this);
2660
2661 // see if we can handle this file; for now we insist it has an ".ovf" extension
2662 Utf8Str strPath = path;
2663 if (!strPath.endsWith(".ovf", Utf8Str::CaseInsensitive))
2664 return setError(VBOX_E_FILE_ERROR,
2665 tr("Appliance file must have .ovf extension"));
2666
2667 ComObjPtr<Progress> progress;
2668 Utf8Str strFormat(format);
2669 TaskWriteOVF::OVFFormat ovfF;
2670 if (strFormat == "ovf-0.9")
2671 ovfF = TaskWriteOVF::OVF_0_9;
2672 else if (strFormat == "ovf-1.0")
2673 ovfF = TaskWriteOVF::OVF_1_0;
2674 else
2675 return setError(VBOX_E_FILE_ERROR,
2676 tr("Invalid format \"%s\" specified"), strFormat.c_str());
2677
2678 rc = writeImpl(ovfF, strPath, progress);
2679
2680 if (SUCCEEDED(rc))
2681 /* Return progress to the caller */
2682 progress.queryInterfaceTo(aProgress);
2683
2684 return rc;
2685}
2686
2687void Appliance::parseURI(Utf8Str strUri, const Utf8Str &strProtocol, Utf8Str &strFilepath, Utf8Str &strHostname, Utf8Str &strUsername, Utf8Str &strPassword)
2688{
2689 /* Remove the protocol */
2690 if (strUri.startsWith(strProtocol, Utf8Str::CaseInsensitive))
2691 strUri = strUri.substr(strProtocol.length());
2692 size_t uppos = strUri.find("@");
2693 if (uppos != Utf8Str::npos)
2694 {
2695 strUsername = strUri.substr(0, uppos);
2696 strUri = strUri.substr(uppos + 1);
2697 size_t upos = strUsername.find(":");
2698 if (upos != Utf8Str::npos)
2699 {
2700 strPassword = strUsername.substr(upos + 1);
2701 strUsername = strUsername.substr(0, upos);
2702 }
2703 }
2704 size_t hpos = strUri.find("/");
2705 if (hpos != Utf8Str::npos)
2706 {
2707 strHostname = strUri.substr(0, hpos);
2708 strUri = strUri.substr(hpos);
2709 }
2710 strFilepath = strUri;
2711}
2712
2713HRESULT Appliance::writeImpl(int aFormat, Utf8Str aPath, ComObjPtr<Progress> &aProgress)
2714{
2715 HRESULT rc = S_OK;
2716 try
2717 {
2718 m->strPath = aPath;
2719
2720 /* Initialize our worker task */
2721 std::auto_ptr<TaskWriteOVF> task(new TaskWriteOVF((TaskWriteOVF::OVFFormat)aFormat, this));
2722
2723 /* Check which kind of export the user has requested */
2724 Utf8Str strProtocol = "";
2725 /* Check the URI for the target format */
2726 if (m->strPath.startsWith("SunCloud://", Utf8Str::CaseInsensitive)) /* Sun Cloud service */
2727 {
2728 throw E_NOTIMPL;
2729// task->storageType = VFSType_S3;
2730// strProtocol = "SunCloud://";
2731 }
2732 else if (m->strPath.startsWith("S3://", Utf8Str::CaseInsensitive)) /* S3 service */
2733 {
2734 throw E_NOTIMPL;
2735// task->storageType = VFSType_S3;
2736// strProtocol = "S3://";
2737 }
2738 else if (m->strPath.startsWith("webdav://", Utf8Str::CaseInsensitive)) /* webdav service */
2739 throw E_NOTIMPL;
2740
2741 parseURI(m->strPath, strProtocol, task->filepath, task->hostname, task->username, task->password);
2742 Bstr progressDesc = BstrFmt(tr("Export appliance '%s'"),
2743 task->filepath.c_str());
2744
2745 /* todo: This progress init stuff should be done a little bit more generic */
2746 if (task->storageType == VFSType_S3)
2747 rc = setUpProgressUpload(aProgress, progressDesc);
2748 else
2749 rc = setUpProgress(aProgress, progressDesc);
2750 if (FAILED(rc)) throw rc;
2751
2752 task->progress = aProgress;
2753
2754 rc = task->startThread();
2755 CheckComRCThrowRC(rc);
2756
2757 /* Don't destruct on success */
2758 task.release();
2759 }
2760 catch (HRESULT aRC)
2761 {
2762 rc = aRC;
2763 }
2764
2765 return rc;
2766}
2767
2768DECLCALLBACK(int) Appliance::taskThreadWriteOVF(RTTHREAD /* aThread */, void *pvUser)
2769{
2770 std::auto_ptr<TaskWriteOVF> task(static_cast<TaskWriteOVF*>(pvUser));
2771 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
2772
2773 Appliance *pAppliance = task->pAppliance;
2774
2775 LogFlowFuncEnter();
2776 LogFlowFunc(("Appliance %p\n", pAppliance));
2777
2778 HRESULT rc = S_OK;
2779
2780 switch(task->taskType)
2781 {
2782 case TaskWriteOVF::Write:
2783 {
2784 if (task->storageType == VFSType_File)
2785 rc = pAppliance->writeFS(task.get());
2786 else if (task->storageType == VFSType_S3)
2787 rc = pAppliance->writeS3(task.get());
2788 break;
2789 }
2790 }
2791
2792 LogFlowFunc(("rc=%Rhrc\n", rc));
2793 LogFlowFuncLeave();
2794
2795 return VINF_SUCCESS;
2796}
2797
2798/**
2799 * Worker thread implementation for Write() (ovf writer).
2800 * @param aThread
2801 * @param pvUser
2802 */
2803/* static */
2804int Appliance::writeFS(TaskWriteOVF *pTask)
2805{
2806 LogFlowFuncEnter();
2807 LogFlowFunc(("Appliance %p\n", this));
2808
2809 AutoCaller autoCaller(this);
2810 CheckComRCReturnRC(autoCaller.rc());
2811
2812 AutoWriteLock appLock(this);
2813
2814 HRESULT rc = S_OK;
2815
2816 try
2817 {
2818 xml::Document doc;
2819 xml::ElementNode *pelmRoot = doc.createRootElement("Envelope");
2820
2821 pelmRoot->setAttribute("ovf:version", (pTask->enFormat == TaskWriteOVF::OVF_1_0) ? "1.0" : "0.9");
2822 pelmRoot->setAttribute("xml:lang", "en-US");
2823
2824 Utf8Str strNamespace = (pTask->enFormat == TaskWriteOVF::OVF_0_9)
2825 ? "http://www.vmware.com/schema/ovf/1/envelope" // 0.9
2826 : "http://schemas.dmtf.org/ovf/envelope/1"; // 1.0
2827 pelmRoot->setAttribute("xmlns", strNamespace);
2828 pelmRoot->setAttribute("xmlns:ovf", strNamespace);
2829
2830// pelmRoot->setAttribute("xmlns:ovfstr", "http://schema.dmtf.org/ovf/strings/1");
2831 pelmRoot->setAttribute("xmlns:rasd", "http://schemas.dmtf.org/wbem/wscim/1/cim-schema/2/CIM_ResourceAllocationSettingData");
2832 pelmRoot->setAttribute("xmlns:vssd", "http://schemas.dmtf.org/wbem/wscim/1/cim-schema/2/CIM_VirtualSystemSettingData");
2833 pelmRoot->setAttribute("xmlns:xsi", "http://www.w3.org/2001/XMLSchema-instance");
2834// pelmRoot->setAttribute("xsi:schemaLocation", "http://schemas.dmtf.org/ovf/envelope/1 ../ovf-envelope.xsd");
2835
2836 // <Envelope>/<References>
2837 xml::ElementNode *pelmReferences = pelmRoot->createChild("References"); // 0.9 and 1.0
2838
2839 /* <Envelope>/<DiskSection>:
2840 <DiskSection>
2841 <Info>List of the virtual disks used in the package</Info>
2842 <Disk ovf:capacity="4294967296" ovf:diskId="lamp" ovf:format="http://www.vmware.com/specifications/vmdk.html#compressed" ovf:populatedSize="1924967692"/>
2843 </DiskSection> */
2844 xml::ElementNode *pelmDiskSection;
2845 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
2846 {
2847 // <Section xsi:type="ovf:DiskSection_Type">
2848 pelmDiskSection = pelmRoot->createChild("Section");
2849 pelmDiskSection->setAttribute("xsi:type", "ovf:DiskSection_Type");
2850 }
2851 else
2852 pelmDiskSection = pelmRoot->createChild("DiskSection");
2853
2854 xml::ElementNode *pelmDiskSectionInfo = pelmDiskSection->createChild("Info");
2855 pelmDiskSectionInfo->addContent("List of the virtual disks used in the package");
2856 // for now, set up a map so we have a list of unique disk names (to make
2857 // sure the same disk name is only added once)
2858 map<Utf8Str, const VirtualSystemDescriptionEntry*> mapDisks;
2859
2860 /* <Envelope>/<NetworkSection>:
2861 <NetworkSection>
2862 <Info>Logical networks used in the package</Info>
2863 <Network ovf:name="VM Network">
2864 <Description>The network that the LAMP Service will be available on</Description>
2865 </Network>
2866 </NetworkSection> */
2867 xml::ElementNode *pelmNetworkSection;
2868 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
2869 {
2870 // <Section xsi:type="ovf:NetworkSection_Type">
2871 pelmNetworkSection = pelmRoot->createChild("Section");
2872 pelmNetworkSection->setAttribute("xsi:type", "ovf:NetworkSection_Type");
2873 }
2874 else
2875 pelmNetworkSection = pelmRoot->createChild("NetworkSection");
2876
2877 xml::ElementNode *pelmNetworkSectionInfo = pelmNetworkSection->createChild("Info");
2878 pelmNetworkSectionInfo->addContent("Logical networks used in the package");
2879 // for now, set up a map so we have a list of unique network names (to make
2880 // sure the same network name is only added once)
2881 map<Utf8Str, bool> mapNetworks;
2882 // we fill this later below when we iterate over the networks
2883
2884 // and here come the virtual systems:
2885
2886 // write a collection if we have more than one virtual system _and_ we're
2887 // writing OVF 1.0; otherwise fail since ovftool can't import more than
2888 // one machine, it seems
2889 xml::ElementNode *pelmToAddVirtualSystemsTo;
2890 if (m->virtualSystemDescriptions.size() > 1)
2891 {
2892 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
2893 throw setError(VBOX_E_FILE_ERROR,
2894 tr("Cannot export more than one virtual system with OVF 0.9, use OVF 1.0"));
2895
2896 pelmToAddVirtualSystemsTo = pelmRoot->createChild("VirtualSystemCollection");
2897 /* xml::AttributeNode *pattrVirtualSystemCollectionId = */ pelmToAddVirtualSystemsTo->setAttribute("ovf:name", "ExportedVirtualBoxMachines"); // whatever
2898 }
2899 else
2900 pelmToAddVirtualSystemsTo = pelmRoot; // add virtual system directly under root element
2901
2902 uint32_t cDisks = 0;
2903
2904 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it;
2905 /* Iterate through all virtual systems of that appliance */
2906 for (it = m->virtualSystemDescriptions.begin();
2907 it != m->virtualSystemDescriptions.end();
2908 ++it)
2909 {
2910 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it);
2911
2912 xml::ElementNode *pelmVirtualSystem;
2913 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
2914 {
2915 // <Section xsi:type="ovf:NetworkSection_Type">
2916 pelmVirtualSystem = pelmToAddVirtualSystemsTo->createChild("Content");
2917 pelmVirtualSystem->setAttribute("xsi:type", "ovf:VirtualSystem_Type");
2918 }
2919 else
2920 pelmVirtualSystem = pelmToAddVirtualSystemsTo->createChild("VirtualSystem");
2921
2922 /*xml::ElementNode *pelmVirtualSystemInfo =*/ pelmVirtualSystem->createChild("Info")->addContent("A virtual machine");
2923
2924 std::list<VirtualSystemDescriptionEntry*> llName = vsdescThis->findByType(VirtualSystemDescriptionType_Name);
2925 if (llName.size() != 1)
2926 throw setError(VBOX_E_NOT_SUPPORTED,
2927 tr("Missing VM name"));
2928 Utf8Str &strVMName = llName.front()->strVbox;
2929 pelmVirtualSystem->setAttribute("ovf:id", strVMName);
2930
2931 // product info
2932 std::list<VirtualSystemDescriptionEntry*> llProduct = vsdescThis->findByType(VirtualSystemDescriptionType_Product);
2933 std::list<VirtualSystemDescriptionEntry*> llProductUrl = vsdescThis->findByType(VirtualSystemDescriptionType_ProductUrl);
2934 std::list<VirtualSystemDescriptionEntry*> llVendor = vsdescThis->findByType(VirtualSystemDescriptionType_Vendor);
2935 std::list<VirtualSystemDescriptionEntry*> llVendorUrl = vsdescThis->findByType(VirtualSystemDescriptionType_VendorUrl);
2936 std::list<VirtualSystemDescriptionEntry*> llVersion = vsdescThis->findByType(VirtualSystemDescriptionType_Version);
2937 bool fProduct = llProduct.size() && !llProduct.front()->strVbox.isEmpty();
2938 bool fProductUrl = llProductUrl.size() && !llProductUrl.front()->strVbox.isEmpty();
2939 bool fVendor = llVendor.size() && !llVendor.front()->strVbox.isEmpty();
2940 bool fVendorUrl = llVendorUrl.size() && !llVendorUrl.front()->strVbox.isEmpty();
2941 bool fVersion = llVersion.size() && !llVersion.front()->strVbox.isEmpty();
2942 if (fProduct ||
2943 fProductUrl ||
2944 fVersion ||
2945 fVendorUrl ||
2946 fVersion)
2947 {
2948 /* <Section ovf:required="false" xsi:type="ovf:ProductSection_Type">
2949 <Info>Meta-information about the installed software</Info>
2950 <Product>VAtest</Product>
2951 <Vendor>SUN Microsystems</Vendor>
2952 <Version>10.0</Version>
2953 <ProductUrl>http://blogs.sun.com/VirtualGuru</ProductUrl>
2954 <VendorUrl>http://www.sun.com</VendorUrl>
2955 </Section> */
2956 xml::ElementNode *pelmAnnotationSection;
2957 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
2958 {
2959 // <Section ovf:required="false" xsi:type="ovf:ProductSection_Type">
2960 pelmAnnotationSection = pelmVirtualSystem->createChild("Section");
2961 pelmAnnotationSection->setAttribute("xsi:type", "ovf:ProductSection_Type");
2962 }
2963 else
2964 pelmAnnotationSection = pelmVirtualSystem->createChild("ProductSection");
2965
2966 pelmAnnotationSection->createChild("Info")->addContent("Meta-information about the installed software");
2967 if (fProduct)
2968 pelmAnnotationSection->createChild("Product")->addContent(llProduct.front()->strVbox);
2969 if (fVendor)
2970 pelmAnnotationSection->createChild("Vendor")->addContent(llVendor.front()->strVbox);
2971 if (fVersion)
2972 pelmAnnotationSection->createChild("Version")->addContent(llVersion.front()->strVbox);
2973 if (fProductUrl)
2974 pelmAnnotationSection->createChild("ProductUrl")->addContent(llProductUrl.front()->strVbox);
2975 if (fVendorUrl)
2976 pelmAnnotationSection->createChild("VendorUrl")->addContent(llVendorUrl.front()->strVbox);
2977 }
2978
2979 // description
2980 std::list<VirtualSystemDescriptionEntry*> llDescription = vsdescThis->findByType(VirtualSystemDescriptionType_Description);
2981 if (llDescription.size() &&
2982 !llDescription.front()->strVbox.isEmpty())
2983 {
2984 /* <Section ovf:required="false" xsi:type="ovf:AnnotationSection_Type">
2985 <Info>A human-readable annotation</Info>
2986 <Annotation>Plan 9</Annotation>
2987 </Section> */
2988 xml::ElementNode *pelmAnnotationSection;
2989 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
2990 {
2991 // <Section ovf:required="false" xsi:type="ovf:AnnotationSection_Type">
2992 pelmAnnotationSection = pelmVirtualSystem->createChild("Section");
2993 pelmAnnotationSection->setAttribute("xsi:type", "ovf:AnnotationSection_Type");
2994 }
2995 else
2996 pelmAnnotationSection = pelmVirtualSystem->createChild("AnnotationSection");
2997
2998 pelmAnnotationSection->createChild("Info")->addContent("A human-readable annotation");
2999 pelmAnnotationSection->createChild("Annotation")->addContent(llDescription.front()->strVbox);
3000 }
3001
3002 // license
3003 std::list<VirtualSystemDescriptionEntry*> llLicense = vsdescThis->findByType(VirtualSystemDescriptionType_License);
3004 if (llLicense.size() &&
3005 !llLicense.front()->strVbox.isEmpty())
3006 {
3007 /* <EulaSection>
3008 <Info ovf:msgid="6">License agreement for the Virtual System.</Info>
3009 <License ovf:msgid="1">License terms can go in here.</License>
3010 </EulaSection> */
3011 xml::ElementNode *pelmEulaSection;
3012 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
3013 {
3014 pelmEulaSection = pelmVirtualSystem->createChild("Section");
3015 pelmEulaSection->setAttribute("xsi:type", "ovf:EulaSection_Type");
3016 }
3017 else
3018 pelmEulaSection = pelmVirtualSystem->createChild("EulaSection");
3019
3020 pelmEulaSection->createChild("Info")->addContent("License agreement for the virtual system");
3021 pelmEulaSection->createChild("License")->addContent(llLicense.front()->strVbox);
3022 }
3023
3024 // operating system
3025 std::list<VirtualSystemDescriptionEntry*> llOS = vsdescThis->findByType(VirtualSystemDescriptionType_OS);
3026 if (llOS.size() != 1)
3027 throw setError(VBOX_E_NOT_SUPPORTED,
3028 tr("Missing OS type"));
3029 /* <OperatingSystemSection ovf:id="82">
3030 <Info>Guest Operating System</Info>
3031 <Description>Linux 2.6.x</Description>
3032 </OperatingSystemSection> */
3033 xml::ElementNode *pelmOperatingSystemSection;
3034 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
3035 {
3036 pelmOperatingSystemSection = pelmVirtualSystem->createChild("Section");
3037 pelmOperatingSystemSection->setAttribute("xsi:type", "ovf:OperatingSystemSection_Type");
3038 }
3039 else
3040 pelmOperatingSystemSection = pelmVirtualSystem->createChild("OperatingSystemSection");
3041
3042 pelmOperatingSystemSection->setAttribute("ovf:id", llOS.front()->strOvf);
3043 pelmOperatingSystemSection->createChild("Info")->addContent("The kind of installed guest operating system");
3044 Utf8Str strOSDesc;
3045 convertCIMOSType2VBoxOSType(strOSDesc, (CIMOSType_T)llOS.front()->strOvf.toInt32(), "");
3046 pelmOperatingSystemSection->createChild("Description")->addContent(strOSDesc);
3047
3048 // <VirtualHardwareSection ovf:id="hw1" ovf:transport="iso">
3049 xml::ElementNode *pelmVirtualHardwareSection;
3050 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
3051 {
3052 // <Section xsi:type="ovf:VirtualHardwareSection_Type">
3053 pelmVirtualHardwareSection = pelmVirtualSystem->createChild("Section");
3054 pelmVirtualHardwareSection->setAttribute("xsi:type", "ovf:VirtualHardwareSection_Type");
3055 }
3056 else
3057 pelmVirtualHardwareSection = pelmVirtualSystem->createChild("VirtualHardwareSection");
3058
3059 pelmVirtualHardwareSection->createChild("Info")->addContent("Virtual hardware requirements for a virtual machine");
3060
3061 /* <System>
3062 <vssd:Description>Description of the virtual hardware section.</vssd:Description>
3063 <vssd:ElementName>vmware</vssd:ElementName>
3064 <vssd:InstanceID>1</vssd:InstanceID>
3065 <vssd:VirtualSystemIdentifier>MyLampService</vssd:VirtualSystemIdentifier>
3066 <vssd:VirtualSystemType>vmx-4</vssd:VirtualSystemType>
3067 </System> */
3068 xml::ElementNode *pelmSystem = pelmVirtualHardwareSection->createChild("System");
3069
3070 pelmSystem->createChild("vssd:ElementName")->addContent("Virtual Hardware Family"); // required OVF 1.0
3071
3072 // <vssd:InstanceId>0</vssd:InstanceId>
3073 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
3074 pelmSystem->createChild("vssd:InstanceId")->addContent("0");
3075 else // capitalization changed...
3076 pelmSystem->createChild("vssd:InstanceID")->addContent("0");
3077
3078 // <vssd:VirtualSystemIdentifier>VAtest</vssd:VirtualSystemIdentifier>
3079 pelmSystem->createChild("vssd:VirtualSystemIdentifier")->addContent(strVMName);
3080 // <vssd:VirtualSystemType>vmx-4</vssd:VirtualSystemType>
3081 const char *pcszHardware = "virtualbox-2.2";
3082 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
3083 // pretend to be vmware compatible then
3084 pcszHardware = "vmx-6";
3085 pelmSystem->createChild("vssd:VirtualSystemType")->addContent(pcszHardware);
3086
3087 // loop thru all description entries twice; once to write out all
3088 // devices _except_ disk images, and a second time to assign the
3089 // disk images; this is because disk images need to reference
3090 // IDE controllers, and we can't know their instance IDs without
3091 // assigning them first
3092
3093 uint32_t idIDEController = 0;
3094 int32_t lIDEControllerIndex = 0;
3095 uint32_t idSATAController = 0;
3096 int32_t lSATAControllerIndex = 0;
3097 uint32_t idSCSIController = 0;
3098 int32_t lSCSIControllerIndex = 0;
3099
3100 uint32_t ulInstanceID = 1;
3101
3102 for (size_t uLoop = 1;
3103 uLoop <= 2;
3104 ++uLoop)
3105 {
3106 int32_t lIndexThis = 0;
3107 list<VirtualSystemDescriptionEntry>::const_iterator itD;
3108 for (itD = vsdescThis->m->llDescriptions.begin();
3109 itD != vsdescThis->m->llDescriptions.end();
3110 ++itD, ++lIndexThis)
3111 {
3112 const VirtualSystemDescriptionEntry &desc = *itD;
3113
3114 OVFResourceType_T type = (OVFResourceType_T)0; // if this becomes != 0 then we do stuff
3115 Utf8Str strResourceSubType;
3116
3117 Utf8Str strDescription; // results in <rasd:Description>...</rasd:Description> block
3118 Utf8Str strCaption; // results in <rasd:Caption>...</rasd:Caption> block
3119
3120 uint32_t ulParent = 0;
3121
3122 int32_t lVirtualQuantity = -1;
3123 Utf8Str strAllocationUnits;
3124
3125 int32_t lAddress = -1;
3126 int32_t lBusNumber = -1;
3127 int32_t lAddressOnParent = -1;
3128
3129 int32_t lAutomaticAllocation = -1; // 0 means "false", 1 means "true"
3130 Utf8Str strConnection; // results in <rasd:Connection>...</rasd:Connection> block
3131 Utf8Str strHostResource;
3132
3133 uint64_t uTemp;
3134
3135 switch (desc.type)
3136 {
3137 case VirtualSystemDescriptionType_CPU:
3138 /* <Item>
3139 <rasd:Caption>1 virtual CPU</rasd:Caption>
3140 <rasd:Description>Number of virtual CPUs</rasd:Description>
3141 <rasd:ElementName>virtual CPU</rasd:ElementName>
3142 <rasd:InstanceID>1</rasd:InstanceID>
3143 <rasd:ResourceType>3</rasd:ResourceType>
3144 <rasd:VirtualQuantity>1</rasd:VirtualQuantity>
3145 </Item> */
3146 if (uLoop == 1)
3147 {
3148 strDescription = "Number of virtual CPUs";
3149 type = OVFResourceType_Processor; // 3
3150 desc.strVbox.toInt(uTemp);
3151 lVirtualQuantity = uTemp;
3152 strCaption = Utf8StrFmt("%d virtual CPU", lVirtualQuantity); // without this ovftool won't eat the item
3153 }
3154 break;
3155
3156 case VirtualSystemDescriptionType_Memory:
3157 /* <Item>
3158 <rasd:AllocationUnits>MegaBytes</rasd:AllocationUnits>
3159 <rasd:Caption>256 MB of memory</rasd:Caption>
3160 <rasd:Description>Memory Size</rasd:Description>
3161 <rasd:ElementName>Memory</rasd:ElementName>
3162 <rasd:InstanceID>2</rasd:InstanceID>
3163 <rasd:ResourceType>4</rasd:ResourceType>
3164 <rasd:VirtualQuantity>256</rasd:VirtualQuantity>
3165 </Item> */
3166 if (uLoop == 1)
3167 {
3168 strDescription = "Memory Size";
3169 type = OVFResourceType_Memory; // 4
3170 desc.strVbox.toInt(uTemp);
3171 lVirtualQuantity = (int32_t)(uTemp / _1M);
3172 strAllocationUnits = "MegaBytes";
3173 strCaption = Utf8StrFmt("%d MB of memory", lVirtualQuantity); // without this ovftool won't eat the item
3174 }
3175 break;
3176
3177 case VirtualSystemDescriptionType_HardDiskControllerIDE:
3178 /* <Item>
3179 <rasd:Caption>ideController1</rasd:Caption>
3180 <rasd:Description>IDE Controller</rasd:Description>
3181 <rasd:InstanceId>5</rasd:InstanceId>
3182 <rasd:ResourceType>5</rasd:ResourceType>
3183 <rasd:Address>1</rasd:Address>
3184 <rasd:BusNumber>1</rasd:BusNumber>
3185 </Item> */
3186 if (uLoop == 1)
3187 {
3188 strDescription = "IDE Controller";
3189 strCaption = "ideController0";
3190 type = OVFResourceType_IDEController; // 5
3191 strResourceSubType = desc.strVbox;
3192 // it seems that OVFTool always writes these two, and since we can only
3193 // have one IDE controller, we'll use this as well
3194 lAddress = 1;
3195 lBusNumber = 1;
3196
3197 // remember this ID
3198 idIDEController = ulInstanceID;
3199 lIDEControllerIndex = lIndexThis;
3200 }
3201 break;
3202
3203 case VirtualSystemDescriptionType_HardDiskControllerSATA:
3204 /* <Item>
3205 <rasd:Caption>sataController0</rasd:Caption>
3206 <rasd:Description>SATA Controller</rasd:Description>
3207 <rasd:InstanceId>4</rasd:InstanceId>
3208 <rasd:ResourceType>20</rasd:ResourceType>
3209 <rasd:ResourceSubType>ahci</rasd:ResourceSubType>
3210 <rasd:Address>0</rasd:Address>
3211 <rasd:BusNumber>0</rasd:BusNumber>
3212 </Item>
3213 */
3214 if (uLoop == 1)
3215 {
3216 strDescription = "SATA Controller";
3217 strCaption = "sataController0";
3218 type = OVFResourceType_OtherStorageDevice; // 20
3219 // it seems that OVFTool always writes these two, and since we can only
3220 // have one SATA controller, we'll use this as well
3221 lAddress = 0;
3222 lBusNumber = 0;
3223
3224 if ( desc.strVbox.isEmpty() // AHCI is the default in VirtualBox
3225 || (!desc.strVbox.compare("ahci", Utf8Str::CaseInsensitive))
3226 )
3227 strResourceSubType = "AHCI";
3228 else
3229 throw setError(VBOX_E_NOT_SUPPORTED,
3230 tr("Invalid config string \"%s\" in SATA controller"), desc.strVbox.c_str());
3231
3232 // remember this ID
3233 idSATAController = ulInstanceID;
3234 lSATAControllerIndex = lIndexThis;
3235 }
3236 break;
3237
3238 case VirtualSystemDescriptionType_HardDiskControllerSCSI:
3239 /* <Item>
3240 <rasd:Caption>scsiController0</rasd:Caption>
3241 <rasd:Description>SCSI Controller</rasd:Description>
3242 <rasd:InstanceId>4</rasd:InstanceId>
3243 <rasd:ResourceType>6</rasd:ResourceType>
3244 <rasd:ResourceSubType>buslogic</rasd:ResourceSubType>
3245 <rasd:Address>0</rasd:Address>
3246 <rasd:BusNumber>0</rasd:BusNumber>
3247 </Item>
3248 */
3249 if (uLoop == 1)
3250 {
3251 strDescription = "SCSI Controller";
3252 strCaption = "scsiController0";
3253 type = OVFResourceType_ParallelSCSIHBA; // 6
3254 // it seems that OVFTool always writes these two, and since we can only
3255 // have one SATA controller, we'll use this as well
3256 lAddress = 0;
3257 lBusNumber = 0;
3258
3259 if ( desc.strVbox.isEmpty() // LsiLogic is the default in VirtualBox
3260 || (!desc.strVbox.compare("lsilogic", Utf8Str::CaseInsensitive))
3261 )
3262 strResourceSubType = "lsilogic";
3263 else if (!desc.strVbox.compare("buslogic", Utf8Str::CaseInsensitive))
3264 strResourceSubType = "buslogic";
3265 else
3266 throw setError(VBOX_E_NOT_SUPPORTED,
3267 tr("Invalid config string \"%s\" in SCSI controller"), desc.strVbox.c_str());
3268
3269 // remember this ID
3270 idSCSIController = ulInstanceID;
3271 lSCSIControllerIndex = lIndexThis;
3272 }
3273 break;
3274
3275 case VirtualSystemDescriptionType_HardDiskImage:
3276 /* <Item>
3277 <rasd:Caption>disk1</rasd:Caption>
3278 <rasd:InstanceId>8</rasd:InstanceId>
3279 <rasd:ResourceType>17</rasd:ResourceType>
3280 <rasd:HostResource>/disk/vmdisk1</rasd:HostResource>
3281 <rasd:Parent>4</rasd:Parent>
3282 <rasd:AddressOnParent>0</rasd:AddressOnParent>
3283 </Item> */
3284 if (uLoop == 2)
3285 {
3286 Utf8Str strDiskID = Utf8StrFmt("vmdisk%RI32", ++cDisks);
3287
3288 strDescription = "Disk Image";
3289 strCaption = Utf8StrFmt("disk%RI32", cDisks); // this is not used for anything else
3290 type = OVFResourceType_HardDisk; // 17
3291
3292 // the following references the "<Disks>" XML block
3293 strHostResource = Utf8StrFmt("/disk/%s", strDiskID.c_str());
3294
3295 // controller=<index>;channel=<c>
3296 size_t pos1 = desc.strExtraConfig.find("controller=");
3297 size_t pos2 = desc.strExtraConfig.find("channel=");
3298 if (pos1 != Utf8Str::npos)
3299 {
3300 int32_t lControllerIndex = -1;
3301 RTStrToInt32Ex(desc.strExtraConfig.c_str() + pos1 + 11, NULL, 0, &lControllerIndex);
3302 if (lControllerIndex == lIDEControllerIndex)
3303 ulParent = idIDEController;
3304 else if (lControllerIndex == lSCSIControllerIndex)
3305 ulParent = idSCSIController;
3306 else if (lControllerIndex == lSATAControllerIndex)
3307 ulParent = idSATAController;
3308 }
3309 if (pos2 != Utf8Str::npos)
3310 RTStrToInt32Ex(desc.strExtraConfig.c_str() + pos2 + 8, NULL, 0, &lAddressOnParent);
3311
3312 if ( !ulParent
3313 || lAddressOnParent == -1
3314 )
3315 throw setError(VBOX_E_NOT_SUPPORTED,
3316 tr("Missing or bad extra config string in hard disk image: \"%s\""), desc.strExtraConfig.c_str());
3317
3318 mapDisks[strDiskID] = &desc;
3319 }
3320 break;
3321
3322 case VirtualSystemDescriptionType_Floppy:
3323 if (uLoop == 1)
3324 {
3325 strDescription = "Floppy Drive";
3326 strCaption = "floppy0"; // this is what OVFTool writes
3327 type = OVFResourceType_FloppyDrive; // 14
3328 lAutomaticAllocation = 0;
3329 lAddressOnParent = 0; // this is what OVFTool writes
3330 }
3331 break;
3332
3333 case VirtualSystemDescriptionType_CDROM:
3334 if (uLoop == 2)
3335 {
3336 // we can't have a CD without an IDE controller
3337 if (!idIDEController)
3338 throw setError(VBOX_E_NOT_SUPPORTED,
3339 tr("Can't have CD-ROM without IDE controller"));
3340
3341 strDescription = "CD-ROM Drive";
3342 strCaption = "cdrom1"; // this is what OVFTool writes
3343 type = OVFResourceType_CDDrive; // 15
3344 lAutomaticAllocation = 1;
3345 ulParent = idIDEController;
3346 lAddressOnParent = 0; // this is what OVFTool writes
3347 }
3348 break;
3349
3350 case VirtualSystemDescriptionType_NetworkAdapter:
3351 /* <Item>
3352 <rasd:AutomaticAllocation>true</rasd:AutomaticAllocation>
3353 <rasd:Caption>Ethernet adapter on 'VM Network'</rasd:Caption>
3354 <rasd:Connection>VM Network</rasd:Connection>
3355 <rasd:ElementName>VM network</rasd:ElementName>
3356 <rasd:InstanceID>3</rasd:InstanceID>
3357 <rasd:ResourceType>10</rasd:ResourceType>
3358 </Item> */
3359 if (uLoop == 1)
3360 {
3361 lAutomaticAllocation = 1;
3362 strCaption = Utf8StrFmt("Ethernet adapter on '%s'", desc.strOvf.c_str());
3363 type = OVFResourceType_EthernetAdapter; // 10
3364 /* Set the hardware type to something useful.
3365 * To be compatible with vmware & others we set
3366 * PCNet32 for our PCNet types & E1000 for the
3367 * E1000 cards. */
3368 switch (desc.strVbox.toInt32())
3369 {
3370 case NetworkAdapterType_Am79C970A:
3371 case NetworkAdapterType_Am79C973: strResourceSubType = "PCNet32"; break;
3372#ifdef VBOX_WITH_E1000
3373 case NetworkAdapterType_I82540EM:
3374 case NetworkAdapterType_I82545EM:
3375 case NetworkAdapterType_I82543GC: strResourceSubType = "E1000"; break;
3376#endif /* VBOX_WITH_E1000 */
3377 }
3378 strConnection = desc.strOvf;
3379
3380 mapNetworks[desc.strOvf] = true;
3381 }
3382 break;
3383
3384 case VirtualSystemDescriptionType_USBController:
3385 /* <Item ovf:required="false">
3386 <rasd:Caption>usb</rasd:Caption>
3387 <rasd:Description>USB Controller</rasd:Description>
3388 <rasd:InstanceId>3</rasd:InstanceId>
3389 <rasd:ResourceType>23</rasd:ResourceType>
3390 <rasd:Address>0</rasd:Address>
3391 <rasd:BusNumber>0</rasd:BusNumber>
3392 </Item> */
3393 if (uLoop == 1)
3394 {
3395 strDescription = "USB Controller";
3396 strCaption = "usb";
3397 type = OVFResourceType_USBController; // 23
3398 lAddress = 0; // this is what OVFTool writes
3399 lBusNumber = 0; // this is what OVFTool writes
3400 }
3401 break;
3402
3403 case VirtualSystemDescriptionType_SoundCard:
3404 /* <Item ovf:required="false">
3405 <rasd:Caption>sound</rasd:Caption>
3406 <rasd:Description>Sound Card</rasd:Description>
3407 <rasd:InstanceId>10</rasd:InstanceId>
3408 <rasd:ResourceType>35</rasd:ResourceType>
3409 <rasd:ResourceSubType>ensoniq1371</rasd:ResourceSubType>
3410 <rasd:AutomaticAllocation>false</rasd:AutomaticAllocation>
3411 <rasd:AddressOnParent>3</rasd:AddressOnParent>
3412 </Item> */
3413 if (uLoop == 1)
3414 {
3415 strDescription = "Sound Card";
3416 strCaption = "sound";
3417 type = OVFResourceType_SoundCard; // 35
3418 strResourceSubType = desc.strOvf; // e.g. ensoniq1371
3419 lAutomaticAllocation = 0;
3420 lAddressOnParent = 3; // what gives? this is what OVFTool writes
3421 }
3422 break;
3423 }
3424
3425 if (type)
3426 {
3427 xml::ElementNode *pItem;
3428
3429 pItem = pelmVirtualHardwareSection->createChild("Item");
3430
3431 // NOTE: do not change the order of these items without good reason! While we don't care
3432 // about ordering, VMware's ovftool does and fails if the items are not written in
3433 // exactly this order, as stupid as it seems.
3434
3435 if (!strCaption.isEmpty())
3436 {
3437 pItem->createChild("rasd:Caption")->addContent(strCaption);
3438 if (pTask->enFormat == TaskWriteOVF::OVF_1_0)
3439 pItem->createChild("rasd:ElementName")->addContent(strCaption);
3440 }
3441
3442 if (!strDescription.isEmpty())
3443 pItem->createChild("rasd:Description")->addContent(strDescription);
3444
3445 // <rasd:InstanceID>1</rasd:InstanceID>
3446 xml::ElementNode *pelmInstanceID;
3447 if (pTask->enFormat == TaskWriteOVF::OVF_0_9)
3448 pelmInstanceID = pItem->createChild("rasd:InstanceId");
3449 else
3450 pelmInstanceID = pItem->createChild("rasd:InstanceID"); // capitalization changed...
3451 pelmInstanceID->addContent(Utf8StrFmt("%d", ulInstanceID++));
3452
3453 // <rasd:ResourceType>3</rasd:ResourceType>
3454 pItem->createChild("rasd:ResourceType")->addContent(Utf8StrFmt("%d", type));
3455 if (!strResourceSubType.isEmpty())
3456 pItem->createChild("rasd:ResourceSubType")->addContent(strResourceSubType);
3457
3458 if (!strHostResource.isEmpty())
3459 pItem->createChild("rasd:HostResource")->addContent(strHostResource);
3460
3461 if (!strAllocationUnits.isEmpty())
3462 pItem->createChild("rasd:AllocationUnits")->addContent(strAllocationUnits);
3463
3464 // <rasd:VirtualQuantity>1</rasd:VirtualQuantity>
3465 if (lVirtualQuantity != -1)
3466 pItem->createChild("rasd:VirtualQuantity")->addContent(Utf8StrFmt("%d", lVirtualQuantity));
3467
3468 if (lAutomaticAllocation != -1)
3469 pItem->createChild("rasd:AutomaticAllocation")->addContent( (lAutomaticAllocation) ? "true" : "false" );
3470
3471 if (!strConnection.isEmpty())
3472 pItem->createChild("rasd:Connection")->addContent(strConnection);
3473
3474 if (lAddress != -1)
3475 pItem->createChild("rasd:Address")->addContent(Utf8StrFmt("%d", lAddress));
3476
3477 if (lBusNumber != -1)
3478 if (pTask->enFormat == TaskWriteOVF::OVF_0_9) // BusNumber is invalid OVF 1.0 so only write it in 0.9 mode for OVFTool compatibility
3479 pItem->createChild("rasd:BusNumber")->addContent(Utf8StrFmt("%d", lBusNumber));
3480
3481 if (ulParent)
3482 pItem->createChild("rasd:Parent")->addContent(Utf8StrFmt("%d", ulParent));
3483 if (lAddressOnParent != -1)
3484 pItem->createChild("rasd:AddressOnParent")->addContent(Utf8StrFmt("%d", lAddressOnParent));
3485 }
3486 }
3487 } // for (size_t uLoop = 0; ...
3488 }
3489
3490 // finally, fill in the network section we set up empty above according
3491 // to the networks we found with the hardware items
3492 map<Utf8Str, bool>::const_iterator itN;
3493 for (itN = mapNetworks.begin();
3494 itN != mapNetworks.end();
3495 ++itN)
3496 {
3497 const Utf8Str &strNetwork = itN->first;
3498 xml::ElementNode *pelmNetwork = pelmNetworkSection->createChild("Network");
3499 pelmNetwork->setAttribute("ovf:name", strNetwork.c_str());
3500 pelmNetwork->createChild("Description")->addContent("Logical network used by this appliance.");
3501 }
3502
3503 map<Utf8Str, const VirtualSystemDescriptionEntry*>::const_iterator itS;
3504 uint32_t ulFile = 1;
3505 for (itS = mapDisks.begin();
3506 itS != mapDisks.end();
3507 ++itS)
3508 {
3509 const Utf8Str &strDiskID = itS->first;
3510 const VirtualSystemDescriptionEntry *pDiskEntry = itS->second;
3511
3512 // source path: where the VBox image is
3513 const Utf8Str &strSrcFilePath = pDiskEntry->strVbox;
3514 Bstr bstrSrcFilePath(strSrcFilePath);
3515 if (!RTPathExists(strSrcFilePath.c_str()))
3516 /* This isn't allowed */
3517 throw setError(VBOX_E_FILE_ERROR,
3518 tr("Source virtual disk image file '%s' doesn't exist"),
3519 strSrcFilePath.c_str());
3520
3521 // output filename
3522 const Utf8Str &strTargetFileNameOnly = pDiskEntry->strOvf;
3523 // target path needs to be composed from where the output OVF is
3524 Utf8Str strTargetFilePath = stripFilename(m->strPath);
3525 strTargetFilePath.append("/");
3526 strTargetFilePath.append(strTargetFileNameOnly);
3527
3528 // clone the disk:
3529 ComPtr<IHardDisk> pSourceDisk;
3530 ComPtr<IHardDisk> pTargetDisk;
3531 ComPtr<IProgress> pProgress2;
3532
3533 Log(("Finding source disk \"%ls\"\n", bstrSrcFilePath.raw()));
3534 rc = mVirtualBox->FindHardDisk(bstrSrcFilePath, pSourceDisk.asOutParam());
3535 if (FAILED(rc)) throw rc;
3536
3537 /* We are always exporting to vmdfk stream optimized for now */
3538 Bstr bstrSrcFormat = L"VMDK";
3539
3540 // create a new hard disk interface for the destination disk image
3541 Log(("Creating target disk \"%s\"\n", strTargetFilePath.raw()));
3542 rc = mVirtualBox->CreateHardDisk(bstrSrcFormat, Bstr(strTargetFilePath), pTargetDisk.asOutParam());
3543 if (FAILED(rc)) throw rc;
3544
3545 // the target disk is now registered and needs to be removed again,
3546 // both after successful cloning or if anything goes bad!
3547 try
3548 {
3549 // create a flat copy of the source disk image
3550 rc = pSourceDisk->CloneTo(pTargetDisk, HardDiskVariant_VmdkStreamOptimized, NULL, pProgress2.asOutParam());
3551 if (FAILED(rc)) throw rc;
3552
3553 // advance to the next operation
3554 if (!pTask->progress.isNull())
3555 pTask->progress->setNextOperation(BstrFmt(tr("Exporting virtual disk image '%s'"), strSrcFilePath.c_str()),
3556 pDiskEntry->ulSizeMB); // operation's weight, as set up with the IProgress originally);
3557
3558 // now wait for the background disk operation to complete; this throws HRESULTs on error
3559 waitForAsyncProgress(pTask->progress, pProgress2);
3560 }
3561 catch (HRESULT rc3)
3562 {
3563 // upon error after registering, close the disk or
3564 // it'll stick in the registry forever
3565 pTargetDisk->Close();
3566 throw;
3567 }
3568
3569 // we need the following for the XML
3570 uint64_t cbFile = 0; // actual file size
3571 rc = pTargetDisk->COMGETTER(Size)(&cbFile);
3572 if (FAILED(rc)) throw rc;
3573
3574 ULONG64 cbCapacity = 0; // size reported to guest
3575 rc = pTargetDisk->COMGETTER(LogicalSize)(&cbCapacity);
3576 if (FAILED(rc)) throw rc;
3577 // capacity is reported in megabytes, so...
3578 cbCapacity *= _1M;
3579
3580 // upon success, close the disk as well
3581 rc = pTargetDisk->Close();
3582 if (FAILED(rc)) throw rc;
3583
3584 // now handle the XML for the disk:
3585 Utf8StrFmt strFileRef("file%RI32", ulFile++);
3586 // <File ovf:href="WindowsXpProfessional-disk1.vmdk" ovf:id="file1" ovf:size="1710381056"/>
3587 xml::ElementNode *pelmFile = pelmReferences->createChild("File");
3588 pelmFile->setAttribute("ovf:href", strTargetFileNameOnly);
3589 pelmFile->setAttribute("ovf:id", strFileRef);
3590 pelmFile->setAttribute("ovf:size", Utf8StrFmt("%RI64", cbFile).c_str());
3591
3592 // add disk to XML Disks section
3593 // <Disk ovf:capacity="8589934592" ovf:diskId="vmdisk1" ovf:fileRef="file1" ovf:format="http://www.vmware.com/specifications/vmdk.html#sparse"/>
3594 xml::ElementNode *pelmDisk = pelmDiskSection->createChild("Disk");
3595 pelmDisk->setAttribute("ovf:capacity", Utf8StrFmt("%RI64", cbCapacity).c_str());
3596 pelmDisk->setAttribute("ovf:diskId", strDiskID);
3597 pelmDisk->setAttribute("ovf:fileRef", strFileRef);
3598 pelmDisk->setAttribute("ovf:format", "http://www.vmware.com/specifications/vmdk.html#sparse"); // must be sparse or ovftool chokes
3599 }
3600
3601 // now go write the XML
3602 xml::XmlFileWriter writer(doc);
3603 writer.write(m->strPath.c_str());
3604 }
3605 catch(xml::Error &x)
3606 {
3607 rc = setError(VBOX_E_FILE_ERROR,
3608 x.what());
3609 }
3610 catch(HRESULT aRC)
3611 {
3612 rc = aRC;
3613 }
3614
3615 pTask->rc = rc;
3616
3617 if (!pTask->progress.isNull())
3618 pTask->progress->notifyComplete(rc);
3619
3620 LogFlowFunc(("rc=%Rhrc\n", rc));
3621 LogFlowFuncLeave();
3622
3623 return VINF_SUCCESS;
3624}
3625
3626/**
3627 * Worker thread implementation for Upload() (ovf uploader).
3628 * @param aThread
3629 * @param pvUser
3630 */
3631/* static */
3632int Appliance::writeS3(TaskWriteOVF *pTask)
3633{
3634 LogFlowFuncEnter();
3635 LogFlowFunc(("Appliance %p\n", this));
3636
3637 AutoCaller autoCaller(this);
3638 CheckComRCReturnRC(autoCaller.rc());
3639
3640 HRESULT rc = S_OK;
3641
3642 AutoWriteLock appLock(this);
3643
3644 /* Buckets are S3 specific. So parse the bucket out of the file path */
3645 Utf8Str tmpPath = pTask->filepath;
3646 if (!tmpPath.startsWith("/"))
3647 return setError(E_INVALIDARG,
3648 tr("The path '%s' must start with /"), tmpPath.c_str());
3649 Utf8Str bucket;
3650 size_t bpos = tmpPath.find("/", 1);
3651 if (bpos != Utf8Str::npos)
3652 {
3653 bucket = tmpPath.substr(1, bpos - 1); /* The bucket without any slashes */
3654 tmpPath = tmpPath.substr(bpos); /* The rest of the file path */
3655 }
3656 /* If there is no bucket name provided reject the upload */
3657 if (bucket.isEmpty())
3658 return setError(E_INVALIDARG,
3659 tr("You doesn't provide a bucket name in the URI"), tmpPath.c_str());
3660
3661 int vrc = VINF_SUCCESS;
3662 RTS3 hS3 = NULL;
3663 char szOSTmpDir[RTPATH_MAX];
3664 RTPathTemp(szOSTmpDir, sizeof(szOSTmpDir));
3665 /* The template for the temporary directory created below */
3666 char *pszTmpDir;
3667 RTStrAPrintf(&pszTmpDir, "%s"RTPATH_SLASH_STR"vbox-ovf-XXXXXX", szOSTmpDir);
3668 list< pair<Utf8Str, ULONG> > filesList;
3669
3670 // todo:
3671 // - getting the tmp directory (especially on win)
3672 // - usable error codes
3673 // - seems snapshot filenames are problematic {uuid}.vdi
3674 try
3675 {
3676 /* We need a temporary directory which we can put the OVF file & all
3677 * disk images in */
3678 vrc = RTDirCreateTemp(pszTmpDir);
3679 if (RT_FAILURE(rc))
3680 throw setError(VBOX_E_FILE_ERROR,
3681 tr("Cannot create temporary directory '%s'"), pszTmpDir);
3682
3683 /* The temporary name of the target OVF file */
3684 Utf8StrFmt strTmpOvf("%s/%s", pszTmpDir, RTPathFilename(tmpPath));
3685
3686 /* Prepare the temporary writing of the OVF */
3687 ComObjPtr<Progress> progress;
3688 rc = writeImpl(pTask->enFormat, strTmpOvf.c_str(), progress);
3689 if (FAILED(rc)) throw rc;
3690
3691 /* Unlock the appliance for the writing thread */
3692 appLock.unlock();
3693 /* Wait until the writing is done, but report the progress back to the
3694 caller */
3695 ComPtr<IProgress> progressInt(progress);
3696 waitForAsyncProgress(pTask->progress, progressInt); /* Any errors will be thrown */
3697
3698 /* Again lock the appliance for the next steps */
3699 appLock.lock();
3700
3701 vrc = RTPathExists(strTmpOvf.c_str()); /* Paranoid check */
3702 if(RT_FAILURE(vrc))
3703 throw setError(VBOX_E_FILE_ERROR,
3704 tr("Cannot find source file '%s'"), strTmpOvf.c_str());
3705 /* Add the OVF file */
3706 filesList.push_back(pair<Utf8Str, ULONG>(strTmpOvf, m->ulWeightPerOperation)); /* Use 1% of the total for the OVF file upload */
3707
3708 /* Now add every disks of every virtual system */
3709 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it;
3710 for (it = m->virtualSystemDescriptions.begin();
3711 it != m->virtualSystemDescriptions.end();
3712 ++it)
3713 {
3714 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it);
3715 std::list<VirtualSystemDescriptionEntry*> avsdeHDs = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskImage);
3716 std::list<VirtualSystemDescriptionEntry*>::const_iterator itH;
3717 for (itH = avsdeHDs.begin();
3718 itH != avsdeHDs.end();
3719 ++itH)
3720 {
3721 const Utf8Str &strTargetFileNameOnly = (*itH)->strOvf;
3722 /* Target path needs to be composed from where the output OVF is */
3723 Utf8Str strTargetFilePath = stripFilename(m->strPath);
3724 strTargetFilePath.append("/");
3725 strTargetFilePath.append(strTargetFileNameOnly);
3726 vrc = RTPathExists(strTargetFilePath.c_str()); /* Paranoid check */
3727 if(RT_FAILURE(vrc))
3728 throw setError(VBOX_E_FILE_ERROR,
3729 tr("Cannot find source file '%s'"), strTargetFilePath.c_str());
3730 filesList.push_back(pair<Utf8Str, ULONG>(strTargetFilePath, (*itH)->ulSizeMB));
3731 }
3732 }
3733 /* Next we have to upload the OVF & all disk images */
3734 vrc = RTS3Create(&hS3, pTask->username.c_str(), pTask->password.c_str(), pTask->hostname.c_str(), "virtualbox-agent/"VBOX_VERSION_STRING);
3735 if(RT_FAILURE(vrc))
3736 throw setError(VBOX_E_IPRT_ERROR,
3737 tr("Cannot create S3 service handler"));
3738 RTS3SetProgressCallback(hS3, pTask->uploadProgress, &pTask);
3739
3740 /* Upload all files */
3741 for (list< pair<Utf8Str, ULONG> >::const_iterator it1 = filesList.begin(); it1 != filesList.end(); ++it1)
3742 {
3743 const pair<Utf8Str, ULONG> &s = (*it1);
3744 char *pszFilename = RTPathFilename(s.first.c_str());
3745 /* Advance to the next operation */
3746 if (!pTask->progress.isNull())
3747 pTask->progress->setNextOperation(BstrFmt(tr("Uploading file '%s'"), pszFilename), s.second);
3748 vrc = RTS3PutKey(hS3, bucket.c_str(), pszFilename, s.first.c_str());
3749 if (RT_FAILURE(vrc))
3750 {
3751 if(vrc == VERR_S3_CANCELED)
3752 break;
3753 else if(vrc == VERR_S3_ACCESS_DENIED)
3754 throw setError(E_ACCESSDENIED,
3755 tr("Cannot upload file '%s' to S3 storage server (Access denied)"), pszFilename);
3756 else if(vrc == VERR_S3_NOT_FOUND)
3757 throw setError(VBOX_E_FILE_ERROR,
3758 tr("Cannot upload file '%s' to S3 storage server (File not found)"), pszFilename);
3759 else
3760 throw setError(VBOX_E_IPRT_ERROR,
3761 tr("Cannot upload file '%s' to S3 storage server (%Rrc)"), pszFilename, vrc);
3762 }
3763 }
3764
3765 }
3766 catch(HRESULT aRC)
3767 {
3768 rc = aRC;
3769 }
3770 /* Cleanup */
3771 if (hS3)
3772 RTS3Destroy(hS3);
3773 /* Delete all files which where temporary created */
3774 for (list< pair<Utf8Str, ULONG> >::const_iterator it1 = filesList.begin(); it1 != filesList.end(); ++it1)
3775 {
3776 const pair<Utf8Str, ULONG> &s = (*it1);
3777 vrc = RTFileDelete(s.first.c_str());
3778 if(RT_FAILURE(vrc))
3779 rc = setError(VBOX_E_FILE_ERROR,
3780 tr("Cannot delete file '%s' (%Rrc)"), s.first.c_str(), vrc);
3781 }
3782 /* Delete the temporary directory */
3783 if (RTPathExists(pszTmpDir))
3784 {
3785 vrc = RTDirRemove(pszTmpDir);
3786 if(RT_FAILURE(vrc))
3787 rc = setError(VBOX_E_FILE_ERROR,
3788 tr("Cannot delete temporary directory '%s' (%Rrc)"), pszTmpDir, vrc);
3789 }
3790 if (pszTmpDir)
3791 RTStrFree(pszTmpDir);
3792
3793 pTask->rc = rc;
3794
3795 if (!pTask->progress.isNull())
3796 pTask->progress->notifyComplete(rc);
3797
3798 LogFlowFunc(("rc=%Rhrc\n", rc));
3799 LogFlowFuncLeave();
3800
3801 return VINF_SUCCESS;
3802}
3803
3804/**
3805* Public method implementation.
3806 * @return
3807 */
3808STDMETHODIMP Appliance::GetWarnings(ComSafeArrayOut(BSTR, aWarnings))
3809{
3810 if (ComSafeArrayOutIsNull(aWarnings))
3811 return E_POINTER;
3812
3813 AutoCaller autoCaller(this);
3814 CheckComRCReturnRC(autoCaller.rc());
3815
3816 AutoReadLock alock(this);
3817
3818 com::SafeArray<BSTR> sfaWarnings(m->llWarnings.size());
3819
3820 list<Utf8Str>::const_iterator it;
3821 size_t i = 0;
3822 for (it = m->llWarnings.begin();
3823 it != m->llWarnings.end();
3824 ++it, ++i)
3825 {
3826 Bstr bstr = *it;
3827 bstr.cloneTo(&sfaWarnings[i]);
3828 }
3829
3830 sfaWarnings.detachTo(ComSafeArrayOutArg(aWarnings));
3831
3832 return S_OK;
3833}
3834
3835HRESULT Appliance::searchUniqueVMName(Utf8Str& aName) const
3836{
3837 IMachine *machine = NULL;
3838 char *tmpName = RTStrDup(aName.c_str());
3839 int i = 1;
3840 /* @todo: Maybe too cost-intensive; try to find a lighter way */
3841 while (mVirtualBox->FindMachine(Bstr(tmpName), &machine) != VBOX_E_OBJECT_NOT_FOUND)
3842 {
3843 RTStrFree(tmpName);
3844 RTStrAPrintf(&tmpName, "%s_%d", aName.c_str(), i);
3845 ++i;
3846 }
3847 aName = tmpName;
3848 RTStrFree(tmpName);
3849
3850 return S_OK;
3851}
3852
3853HRESULT Appliance::searchUniqueDiskImageFilePath(Utf8Str& aName) const
3854{
3855 IHardDisk *harddisk = NULL;
3856 char *tmpName = RTStrDup(aName.c_str());
3857 int i = 1;
3858 /* Check if the file exists or if a file with this path is registered
3859 * already */
3860 /* @todo: Maybe too cost-intensive; try to find a lighter way */
3861 while (RTPathExists(tmpName) ||
3862 mVirtualBox->FindHardDisk(Bstr(tmpName), &harddisk) != VBOX_E_OBJECT_NOT_FOUND)
3863 {
3864 RTStrFree(tmpName);
3865 char *tmpDir = RTStrDup(aName.c_str());
3866 RTPathStripFilename(tmpDir);;
3867 char *tmpFile = RTStrDup(RTPathFilename(aName.c_str()));
3868 RTPathStripExt(tmpFile);
3869 const char *tmpExt = RTPathExt(aName.c_str());
3870 RTStrAPrintf(&tmpName, "%s%c%s_%d%s", tmpDir, RTPATH_DELIMITER, tmpFile, i, tmpExt);
3871 RTStrFree(tmpFile);
3872 RTStrFree(tmpDir);
3873 ++i;
3874 }
3875 aName = tmpName;
3876 RTStrFree(tmpName);
3877
3878 return S_OK;
3879}
3880
3881/**
3882 * Sets up the given progress object so that it represents disk images accurately
3883 * during importMachines() and write().
3884 * @param pProgress
3885 * @param bstrDescription
3886 * @return
3887 */
3888HRESULT Appliance::setUpProgress(ComObjPtr<Progress> &pProgress, const Bstr &bstrDescription)
3889{
3890 HRESULT rc;
3891
3892 /* Create the progress object */
3893 pProgress.createObject();
3894
3895 // weigh the disk images according to their sizes
3896 uint32_t ulTotalMB = 0;
3897 uint32_t cDisks = 0;
3898 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it;
3899 for (it = m->virtualSystemDescriptions.begin();
3900 it != m->virtualSystemDescriptions.end();
3901 ++it)
3902 {
3903 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it);
3904 /* One for every hard disk of the Virtual System */
3905 std::list<VirtualSystemDescriptionEntry*> avsdeHDs = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskImage);
3906 std::list<VirtualSystemDescriptionEntry*>::const_iterator itH;
3907 for (itH = avsdeHDs.begin();
3908 itH != avsdeHDs.end();
3909 ++itH)
3910 {
3911 const VirtualSystemDescriptionEntry *pHD = *itH;
3912 ulTotalMB += pHD->ulSizeMB;
3913 ++cDisks;
3914 }
3915 }
3916
3917 ULONG cOperations = 1 + cDisks; // one op per disk plus 1 for the XML
3918
3919 ULONG ulTotalOperationsWeight;
3920 if (ulTotalMB)
3921 {
3922 m->ulWeightPerOperation = (ULONG)((double)ulTotalMB * 1 / 100); // use 1% of the progress for the XML
3923 ulTotalOperationsWeight = ulTotalMB + m->ulWeightPerOperation;
3924 }
3925 else
3926 {
3927 // no disks to export:
3928 ulTotalOperationsWeight = 1;
3929 m->ulWeightPerOperation = 1;
3930 }
3931
3932 Log(("Setting up progress object: ulTotalMB = %d, cDisks = %d, => cOperations = %d, ulTotalOperationsWeight = %d, m->ulWeightPerOperation = %d\n",
3933 ulTotalMB, cDisks, cOperations, ulTotalOperationsWeight, m->ulWeightPerOperation));
3934
3935 rc = pProgress->init(mVirtualBox, static_cast<IAppliance*>(this),
3936 bstrDescription,
3937 TRUE /* aCancelable */,
3938 cOperations, // ULONG cOperations,
3939 ulTotalOperationsWeight, // ULONG ulTotalOperationsWeight,
3940 bstrDescription, // CBSTR bstrFirstOperationDescription,
3941 m->ulWeightPerOperation); // ULONG ulFirstOperationWeight,
3942 return rc;
3943}
3944
3945HRESULT Appliance::setUpProgressUpload(ComObjPtr<Progress> &pProgress, const Bstr &bstrDescription)
3946{
3947 HRESULT rc;
3948
3949 /* Create the progress object */
3950 pProgress.createObject();
3951
3952 // weigh the disk images according to their sizes
3953 uint32_t ulTotalMB = 0;
3954 uint32_t cDisks = 0;
3955 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it;
3956 for (it = m->virtualSystemDescriptions.begin();
3957 it != m->virtualSystemDescriptions.end();
3958 ++it)
3959 {
3960 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it);
3961 /* One for every hard disk of the Virtual System */
3962 std::list<VirtualSystemDescriptionEntry*> avsdeHDs = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskImage);
3963 std::list<VirtualSystemDescriptionEntry*>::const_iterator itH;
3964 for (itH = avsdeHDs.begin();
3965 itH != avsdeHDs.end();
3966 ++itH)
3967 {
3968 const VirtualSystemDescriptionEntry *pHD = *itH;
3969 ulTotalMB += pHD->ulSizeMB;
3970 ++cDisks;
3971 }
3972 }
3973
3974 ULONG cOperations = 1 + 1 + cDisks; // one op per disk plus 1 for the OVF & 1 plus to the temporary creation */
3975
3976 ULONG ulTotalOperationsWeight;
3977 if (ulTotalMB)
3978 {
3979 m->ulWeightPerOperation = (ULONG)((double)ulTotalMB * 1 / 100); // use 1% of the progress for OVF file upload (we didn't know the size at this point)
3980 ulTotalOperationsWeight = ulTotalMB + m->ulWeightPerOperation;
3981 }
3982 else
3983 {
3984 // no disks to export:
3985 ulTotalOperationsWeight = 1;
3986 m->ulWeightPerOperation = 1;
3987 }
3988 ULONG ulOVFCreationWeight = (ULONG)((double)ulTotalOperationsWeight * 50.0 / 100.0); /* Use 50% for the creation of the OVF & the disks */
3989 ulTotalOperationsWeight += ulOVFCreationWeight;
3990
3991 Log(("Setting up progress object: ulTotalMB = %d, cDisks = %d, => cOperations = %d, ulTotalOperationsWeight = %d, m->ulWeightPerOperation = %d\n",
3992 ulTotalMB, cDisks, cOperations, ulTotalOperationsWeight, m->ulWeightPerOperation));
3993
3994 rc = pProgress->init(mVirtualBox, static_cast<IAppliance*>(this),
3995 bstrDescription,
3996 TRUE /* aCancelable */,
3997 cOperations, // ULONG cOperations,
3998 ulTotalOperationsWeight, // ULONG ulTotalOperationsWeight,
3999 bstrDescription, // CBSTR bstrFirstOperationDescription,
4000 ulOVFCreationWeight); // ULONG ulFirstOperationWeight,
4001 return rc;
4002}
4003
4004/**
4005 * Called from the import and export background threads to synchronize the second
4006 * background disk thread's progress object with the current progress object so
4007 * that the user interface sees progress correctly and that cancel signals are
4008 * passed on to the second thread.
4009 * @param pProgressThis Progress object of the current thread.
4010 * @param pProgressAsync Progress object of asynchronous task running in background.
4011 */
4012void Appliance::waitForAsyncProgress(ComObjPtr<Progress> &pProgressThis,
4013 ComPtr<IProgress> &pProgressAsync)
4014{
4015 HRESULT rc;
4016
4017 // now loop until the asynchronous operation completes and then report its result
4018 BOOL fCompleted;
4019 BOOL fCanceled;
4020 ULONG currentPercent;
4021 while (SUCCEEDED(pProgressAsync->COMGETTER(Completed(&fCompleted))))
4022 {
4023 rc = pProgressThis->COMGETTER(Canceled)(&fCanceled);
4024 if (FAILED(rc)) throw rc;
4025 if (fCanceled)
4026 {
4027 pProgressAsync->Cancel();
4028 break;
4029 }
4030
4031 rc = pProgressAsync->COMGETTER(Percent(&currentPercent));
4032 if (FAILED(rc)) throw rc;
4033 if (!pProgressThis.isNull())
4034 pProgressThis->setCurrentOperationProgress(currentPercent);
4035 if (fCompleted)
4036 break;
4037
4038 /* Make sure the loop is not too tight */
4039 rc = pProgressAsync->WaitForCompletion(100);
4040 if (FAILED(rc)) throw rc;
4041 }
4042 // report result of asynchronous operation
4043 LONG iRc;
4044 rc = pProgressAsync->COMGETTER(ResultCode)(&iRc);
4045 if (FAILED(rc)) throw rc;
4046
4047
4048 // if the thread of the progress object has an error, then
4049 // retrieve the error info from there, or it'll be lost
4050 if (FAILED(iRc))
4051 {
4052 ProgressErrorInfo info(pProgressAsync);
4053 Utf8Str str(info.getText());
4054 const char *pcsz = str.c_str();
4055 HRESULT rc2 = setError(iRc, pcsz);
4056 throw rc2;
4057 }
4058}
4059
4060void Appliance::addWarning(const char* aWarning, ...)
4061{
4062 va_list args;
4063 va_start(args, aWarning);
4064 Utf8StrFmtVA str(aWarning, args);
4065 va_end(args);
4066 m->llWarnings.push_back(str);
4067}
4068
4069////////////////////////////////////////////////////////////////////////////////
4070//
4071// IVirtualSystemDescription constructor / destructor
4072//
4073////////////////////////////////////////////////////////////////////////////////
4074
4075DEFINE_EMPTY_CTOR_DTOR(VirtualSystemDescription)
4076struct shutup3 {};
4077
4078/**
4079 * COM initializer.
4080 * @return
4081 */
4082HRESULT VirtualSystemDescription::init()
4083{
4084 /* Enclose the state transition NotReady->InInit->Ready */
4085 AutoInitSpan autoInitSpan(this);
4086 AssertReturn(autoInitSpan.isOk(), E_FAIL);
4087
4088 /* Initialize data */
4089 m = new Data();
4090
4091 /* Confirm a successful initialization */
4092 autoInitSpan.setSucceeded();
4093 return S_OK;
4094}
4095
4096/**
4097* COM uninitializer.
4098*/
4099
4100void VirtualSystemDescription::uninit()
4101{
4102 delete m;
4103 m = NULL;
4104}
4105
4106////////////////////////////////////////////////////////////////////////////////
4107//
4108// IVirtualSystemDescription public methods
4109//
4110////////////////////////////////////////////////////////////////////////////////
4111
4112/**
4113 * Public method implementation.
4114 * @param
4115 * @return
4116 */
4117STDMETHODIMP VirtualSystemDescription::COMGETTER(Count)(ULONG *aCount)
4118{
4119 if (!aCount)
4120 return E_POINTER;
4121
4122 AutoCaller autoCaller(this);
4123 CheckComRCReturnRC(autoCaller.rc());
4124
4125 AutoReadLock alock(this);
4126
4127 *aCount = (ULONG)m->llDescriptions.size();
4128
4129 return S_OK;
4130}
4131
4132/**
4133 * Public method implementation.
4134 * @return
4135 */
4136STDMETHODIMP VirtualSystemDescription::GetDescription(ComSafeArrayOut(VirtualSystemDescriptionType_T, aTypes),
4137 ComSafeArrayOut(BSTR, aRefs),
4138 ComSafeArrayOut(BSTR, aOrigValues),
4139 ComSafeArrayOut(BSTR, aVboxValues),
4140 ComSafeArrayOut(BSTR, aExtraConfigValues))
4141{
4142 if (ComSafeArrayOutIsNull(aTypes) ||
4143 ComSafeArrayOutIsNull(aRefs) ||
4144 ComSafeArrayOutIsNull(aOrigValues) ||
4145 ComSafeArrayOutIsNull(aVboxValues) ||
4146 ComSafeArrayOutIsNull(aExtraConfigValues))
4147 return E_POINTER;
4148
4149 AutoCaller autoCaller(this);
4150 CheckComRCReturnRC(autoCaller.rc());
4151
4152 AutoReadLock alock(this);
4153
4154 ULONG c = (ULONG)m->llDescriptions.size();
4155 com::SafeArray<VirtualSystemDescriptionType_T> sfaTypes(c);
4156 com::SafeArray<BSTR> sfaRefs(c);
4157 com::SafeArray<BSTR> sfaOrigValues(c);
4158 com::SafeArray<BSTR> sfaVboxValues(c);
4159 com::SafeArray<BSTR> sfaExtraConfigValues(c);
4160
4161 list<VirtualSystemDescriptionEntry>::const_iterator it;
4162 size_t i = 0;
4163 for (it = m->llDescriptions.begin();
4164 it != m->llDescriptions.end();
4165 ++it, ++i)
4166 {
4167 const VirtualSystemDescriptionEntry &vsde = (*it);
4168
4169 sfaTypes[i] = vsde.type;
4170
4171 Bstr bstr = vsde.strRef;
4172 bstr.cloneTo(&sfaRefs[i]);
4173
4174 bstr = vsde.strOvf;
4175 bstr.cloneTo(&sfaOrigValues[i]);
4176
4177 bstr = vsde.strVbox;
4178 bstr.cloneTo(&sfaVboxValues[i]);
4179
4180 bstr = vsde.strExtraConfig;
4181 bstr.cloneTo(&sfaExtraConfigValues[i]);
4182 }
4183
4184 sfaTypes.detachTo(ComSafeArrayOutArg(aTypes));
4185 sfaRefs.detachTo(ComSafeArrayOutArg(aRefs));
4186 sfaOrigValues.detachTo(ComSafeArrayOutArg(aOrigValues));
4187 sfaVboxValues.detachTo(ComSafeArrayOutArg(aVboxValues));
4188 sfaExtraConfigValues.detachTo(ComSafeArrayOutArg(aExtraConfigValues));
4189
4190 return S_OK;
4191}
4192
4193/**
4194 * Public method implementation.
4195 * @return
4196 */
4197STDMETHODIMP VirtualSystemDescription::GetDescriptionByType(VirtualSystemDescriptionType_T aType,
4198 ComSafeArrayOut(VirtualSystemDescriptionType_T, aTypes),
4199 ComSafeArrayOut(BSTR, aRefs),
4200 ComSafeArrayOut(BSTR, aOrigValues),
4201 ComSafeArrayOut(BSTR, aVboxValues),
4202 ComSafeArrayOut(BSTR, aExtraConfigValues))
4203{
4204 if (ComSafeArrayOutIsNull(aTypes) ||
4205 ComSafeArrayOutIsNull(aRefs) ||
4206 ComSafeArrayOutIsNull(aOrigValues) ||
4207 ComSafeArrayOutIsNull(aVboxValues) ||
4208 ComSafeArrayOutIsNull(aExtraConfigValues))
4209 return E_POINTER;
4210
4211 AutoCaller autoCaller(this);
4212 CheckComRCReturnRC(autoCaller.rc());
4213
4214 AutoReadLock alock(this);
4215
4216 std::list<VirtualSystemDescriptionEntry*> vsd = findByType (aType);
4217 ULONG c = (ULONG)vsd.size();
4218 com::SafeArray<VirtualSystemDescriptionType_T> sfaTypes(c);
4219 com::SafeArray<BSTR> sfaRefs(c);
4220 com::SafeArray<BSTR> sfaOrigValues(c);
4221 com::SafeArray<BSTR> sfaVboxValues(c);
4222 com::SafeArray<BSTR> sfaExtraConfigValues(c);
4223
4224 list<VirtualSystemDescriptionEntry*>::const_iterator it;
4225 size_t i = 0;
4226 for (it = vsd.begin();
4227 it != vsd.end();
4228 ++it, ++i)
4229 {
4230 const VirtualSystemDescriptionEntry *vsde = (*it);
4231
4232 sfaTypes[i] = vsde->type;
4233
4234 Bstr bstr = vsde->strRef;
4235 bstr.cloneTo(&sfaRefs[i]);
4236
4237 bstr = vsde->strOvf;
4238 bstr.cloneTo(&sfaOrigValues[i]);
4239
4240 bstr = vsde->strVbox;
4241 bstr.cloneTo(&sfaVboxValues[i]);
4242
4243 bstr = vsde->strExtraConfig;
4244 bstr.cloneTo(&sfaExtraConfigValues[i]);
4245 }
4246
4247 sfaTypes.detachTo(ComSafeArrayOutArg(aTypes));
4248 sfaRefs.detachTo(ComSafeArrayOutArg(aRefs));
4249 sfaOrigValues.detachTo(ComSafeArrayOutArg(aOrigValues));
4250 sfaVboxValues.detachTo(ComSafeArrayOutArg(aVboxValues));
4251 sfaExtraConfigValues.detachTo(ComSafeArrayOutArg(aExtraConfigValues));
4252
4253 return S_OK;
4254}
4255
4256/**
4257 * Public method implementation.
4258 * @return
4259 */
4260STDMETHODIMP VirtualSystemDescription::GetValuesByType(VirtualSystemDescriptionType_T aType,
4261 VirtualSystemDescriptionValueType_T aWhich,
4262 ComSafeArrayOut(BSTR, aValues))
4263{
4264 if (ComSafeArrayOutIsNull(aValues))
4265 return E_POINTER;
4266
4267 AutoCaller autoCaller(this);
4268 CheckComRCReturnRC(autoCaller.rc());
4269
4270 AutoReadLock alock(this);
4271
4272 std::list<VirtualSystemDescriptionEntry*> vsd = findByType (aType);
4273 com::SafeArray<BSTR> sfaValues((ULONG)vsd.size());
4274
4275 list<VirtualSystemDescriptionEntry*>::const_iterator it;
4276 size_t i = 0;
4277 for (it = vsd.begin();
4278 it != vsd.end();
4279 ++it, ++i)
4280 {
4281 const VirtualSystemDescriptionEntry *vsde = (*it);
4282
4283 Bstr bstr;
4284 switch (aWhich)
4285 {
4286 case VirtualSystemDescriptionValueType_Reference: bstr = vsde->strRef; break;
4287 case VirtualSystemDescriptionValueType_Original: bstr = vsde->strOvf; break;
4288 case VirtualSystemDescriptionValueType_Auto: bstr = vsde->strVbox; break;
4289 case VirtualSystemDescriptionValueType_ExtraConfig: bstr = vsde->strExtraConfig; break;
4290 }
4291
4292 bstr.cloneTo(&sfaValues[i]);
4293 }
4294
4295 sfaValues.detachTo(ComSafeArrayOutArg(aValues));
4296
4297 return S_OK;
4298}
4299
4300/**
4301 * Public method implementation.
4302 * @return
4303 */
4304STDMETHODIMP VirtualSystemDescription::SetFinalValues(ComSafeArrayIn(BOOL, aEnabled),
4305 ComSafeArrayIn(IN_BSTR, argVboxValues),
4306 ComSafeArrayIn(IN_BSTR, argExtraConfigValues))
4307{
4308#ifndef RT_OS_WINDOWS
4309 NOREF(aEnabledSize);
4310#endif /* RT_OS_WINDOWS */
4311
4312 CheckComArgSafeArrayNotNull(aEnabled);
4313 CheckComArgSafeArrayNotNull(argVboxValues);
4314 CheckComArgSafeArrayNotNull(argExtraConfigValues);
4315
4316 AutoCaller autoCaller(this);
4317 CheckComRCReturnRC(autoCaller.rc());
4318
4319 AutoWriteLock alock(this);
4320
4321 com::SafeArray<BOOL> sfaEnabled(ComSafeArrayInArg(aEnabled));
4322 com::SafeArray<IN_BSTR> sfaVboxValues(ComSafeArrayInArg(argVboxValues));
4323 com::SafeArray<IN_BSTR> sfaExtraConfigValues(ComSafeArrayInArg(argExtraConfigValues));
4324
4325 if ( (sfaEnabled.size() != m->llDescriptions.size())
4326 || (sfaVboxValues.size() != m->llDescriptions.size())
4327 || (sfaExtraConfigValues.size() != m->llDescriptions.size())
4328 )
4329 return E_INVALIDARG;
4330
4331 list<VirtualSystemDescriptionEntry>::iterator it;
4332 size_t i = 0;
4333 for (it = m->llDescriptions.begin();
4334 it != m->llDescriptions.end();
4335 ++it, ++i)
4336 {
4337 VirtualSystemDescriptionEntry& vsde = *it;
4338
4339 if (sfaEnabled[i])
4340 {
4341 vsde.strVbox = sfaVboxValues[i];
4342 vsde.strExtraConfig = sfaExtraConfigValues[i];
4343 }
4344 else
4345 vsde.type = VirtualSystemDescriptionType_Ignore;
4346 }
4347
4348 return S_OK;
4349}
4350
4351/**
4352 * Public method implementation.
4353 * @return
4354 */
4355STDMETHODIMP VirtualSystemDescription::AddDescription(VirtualSystemDescriptionType_T aType,
4356 IN_BSTR aVboxValue,
4357 IN_BSTR aExtraConfigValue)
4358{
4359 CheckComArgNotNull(aVboxValue);
4360 CheckComArgNotNull(aExtraConfigValue);
4361
4362 AutoCaller autoCaller(this);
4363 CheckComRCReturnRC(autoCaller.rc());
4364
4365 AutoWriteLock alock(this);
4366
4367 addEntry(aType, "", aVboxValue, aVboxValue, 0, aExtraConfigValue);
4368
4369 return S_OK;
4370}
4371
4372/**
4373 * Internal method; adds a new description item to the member list.
4374 * @param aType Type of description for the new item.
4375 * @param strRef Reference item; only used with hard disk controllers.
4376 * @param aOrigValue Corresponding original value from OVF.
4377 * @param aAutoValue Initial configuration value (can be overridden by caller with setFinalValues).
4378 * @param ulSizeMB Weight for IProgress
4379 * @param strExtraConfig Extra configuration; meaning dependent on type.
4380 */
4381void VirtualSystemDescription::addEntry(VirtualSystemDescriptionType_T aType,
4382 const Utf8Str &strRef,
4383 const Utf8Str &aOrigValue,
4384 const Utf8Str &aAutoValue,
4385 uint32_t ulSizeMB,
4386 const Utf8Str &strExtraConfig /*= ""*/)
4387{
4388 VirtualSystemDescriptionEntry vsde;
4389 vsde.ulIndex = (uint32_t)m->llDescriptions.size(); // each entry gets an index so the client side can reference them
4390 vsde.type = aType;
4391 vsde.strRef = strRef;
4392 vsde.strOvf = aOrigValue;
4393 vsde.strVbox = aAutoValue;
4394 vsde.strExtraConfig = strExtraConfig;
4395 vsde.ulSizeMB = ulSizeMB;
4396
4397 m->llDescriptions.push_back(vsde);
4398}
4399
4400/**
4401 * Private method; returns a list of description items containing all the items from the member
4402 * description items of this virtual system that match the given type.
4403 * @param aType
4404 * @return
4405 */
4406std::list<VirtualSystemDescriptionEntry*> VirtualSystemDescription::findByType(VirtualSystemDescriptionType_T aType)
4407{
4408 std::list<VirtualSystemDescriptionEntry*> vsd;
4409
4410 list<VirtualSystemDescriptionEntry>::iterator it;
4411 for (it = m->llDescriptions.begin();
4412 it != m->llDescriptions.end();
4413 ++it)
4414 {
4415 if (it->type == aType)
4416 vsd.push_back(&(*it));
4417 }
4418
4419 return vsd;
4420}
4421
4422/**
4423 * Private method; looks thru the member hardware items for the IDE, SATA, or SCSI controller with
4424 * the given reference ID. Useful when needing the controller for a particular
4425 * virtual disk.
4426 * @param id
4427 * @return
4428 */
4429const VirtualSystemDescriptionEntry* VirtualSystemDescription::findControllerFromID(uint32_t id)
4430{
4431 Utf8Str strRef = Utf8StrFmt("%RI32", id);
4432 list<VirtualSystemDescriptionEntry>::const_iterator it;
4433 for (it = m->llDescriptions.begin();
4434 it != m->llDescriptions.end();
4435 ++it)
4436 {
4437 const VirtualSystemDescriptionEntry &d = *it;
4438 switch (d.type)
4439 {
4440 case VirtualSystemDescriptionType_HardDiskControllerIDE:
4441 case VirtualSystemDescriptionType_HardDiskControllerSATA:
4442 case VirtualSystemDescriptionType_HardDiskControllerSCSI:
4443 if (d.strRef == strRef)
4444 return &d;
4445 break;
4446 }
4447 }
4448
4449 return NULL;
4450}
4451
4452////////////////////////////////////////////////////////////////////////////////
4453//
4454// IMachine public methods
4455//
4456////////////////////////////////////////////////////////////////////////////////
4457
4458// This code is here so we won't have to include the appliance headers in the
4459// IMachine implementation, and we also need to access private appliance data.
4460
4461/**
4462* Public method implementation.
4463* @param appliance
4464* @return
4465*/
4466
4467STDMETHODIMP Machine::Export(IAppliance *aAppliance, IVirtualSystemDescription **aDescription)
4468{
4469 HRESULT rc = S_OK;
4470
4471 if (!aAppliance)
4472 return E_POINTER;
4473
4474 AutoCaller autoCaller(this);
4475 CheckComRCReturnRC(autoCaller.rc());
4476
4477 AutoReadLock alock(this);
4478
4479 ComObjPtr<VirtualSystemDescription> pNewDesc;
4480
4481 try
4482 {
4483 Bstr bstrName;
4484 Bstr bstrDescription;
4485 Bstr bstrGuestOSType;
4486 uint32_t cCPUs;
4487 uint32_t ulMemSizeMB;
4488 BOOL fDVDEnabled;
4489 BOOL fFloppyEnabled;
4490 BOOL fUSBEnabled;
4491 BOOL fAudioEnabled;
4492 AudioControllerType_T audioController;
4493
4494 ComPtr<IUSBController> pUsbController;
4495 ComPtr<IAudioAdapter> pAudioAdapter;
4496
4497 // get name
4498 bstrName = mUserData->mName;
4499 // get description
4500 bstrDescription = mUserData->mDescription;
4501 // get guest OS
4502 bstrGuestOSType = mUserData->mOSTypeId;
4503 // CPU count
4504 cCPUs = mHWData->mCPUCount;
4505 // memory size in MB
4506 ulMemSizeMB = mHWData->mMemorySize;
4507 // VRAM size?
4508 // BIOS settings?
4509 // 3D acceleration enabled?
4510 // hardware virtualization enabled?
4511 // nested paging enabled?
4512 // HWVirtExVPIDEnabled?
4513 // PAEEnabled?
4514 // snapshotFolder?
4515 // VRDPServer?
4516
4517 // floppy
4518 rc = mFloppyDrive->COMGETTER(Enabled)(&fFloppyEnabled);
4519 if (FAILED(rc)) throw rc;
4520
4521 // CD-ROM ?!?
4522 // ComPtr<IDVDDrive> pDVDDrive;
4523 fDVDEnabled = 1;
4524
4525 // this is more tricky so use the COM method
4526 rc = COMGETTER(USBController)(pUsbController.asOutParam());
4527 if (FAILED(rc))
4528 fUSBEnabled = false;
4529 else
4530 rc = pUsbController->COMGETTER(Enabled)(&fUSBEnabled);
4531
4532 pAudioAdapter = mAudioAdapter;
4533 rc = pAudioAdapter->COMGETTER(Enabled)(&fAudioEnabled);
4534 if (FAILED(rc)) throw rc;
4535 rc = pAudioAdapter->COMGETTER(AudioController)(&audioController);
4536 if (FAILED(rc)) throw rc;
4537
4538 // create a new virtual system
4539 rc = pNewDesc.createObject();
4540 CheckComRCThrowRC(rc);
4541 rc = pNewDesc->init();
4542 CheckComRCThrowRC(rc);
4543
4544 /* Guest OS type */
4545 Utf8Str strOsTypeVBox(bstrGuestOSType);
4546 CIMOSType_T cim = convertVBoxOSType2CIMOSType(strOsTypeVBox.c_str());
4547 pNewDesc->addEntry(VirtualSystemDescriptionType_OS,
4548 "",
4549 Utf8StrFmt("%RI32", cim),
4550 strOsTypeVBox);
4551
4552 /* VM name */
4553 Utf8Str strVMName(bstrName);
4554 pNewDesc->addEntry(VirtualSystemDescriptionType_Name,
4555 "",
4556 strVMName,
4557 strVMName);
4558
4559 // description
4560 Utf8Str strDescription(bstrDescription);
4561 pNewDesc->addEntry(VirtualSystemDescriptionType_Description,
4562 "",
4563 strDescription,
4564 strDescription);
4565
4566 /* CPU count*/
4567 Utf8Str strCpuCount = Utf8StrFmt("%RI32", cCPUs);
4568 pNewDesc->addEntry(VirtualSystemDescriptionType_CPU,
4569 "",
4570 strCpuCount,
4571 strCpuCount);
4572
4573 /* Memory */
4574 Utf8Str strMemory = Utf8StrFmt("%RI32", (uint64_t)ulMemSizeMB * _1M);
4575 pNewDesc->addEntry(VirtualSystemDescriptionType_Memory,
4576 "",
4577 strMemory,
4578 strMemory);
4579
4580 int32_t lIDEControllerIndex = 0;
4581 int32_t lSATAControllerIndex = 0;
4582 int32_t lSCSIControllerIndex = 0;
4583
4584// <const name="HardDiskControllerIDE" value="6" />
4585 ComPtr<IStorageController> pController;
4586 rc = GetStorageControllerByName(Bstr("IDE"), pController.asOutParam());
4587 if (FAILED(rc)) throw rc;
4588 Utf8Str strVbox;
4589 StorageControllerType_T ctlr;
4590 rc = pController->COMGETTER(ControllerType)(&ctlr);
4591 if (FAILED(rc)) throw rc;
4592 switch(ctlr)
4593 {
4594 case StorageControllerType_PIIX3: strVbox = "PIIX3"; break;
4595 case StorageControllerType_PIIX4: strVbox = "PIIX4"; break;
4596 case StorageControllerType_ICH6: strVbox = "ICH6"; break;
4597 }
4598
4599 if (strVbox.length())
4600 {
4601 lIDEControllerIndex = (int32_t)pNewDesc->m->llDescriptions.size();
4602 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerIDE,
4603 Utf8StrFmt("%d", lIDEControllerIndex),
4604 strVbox,
4605 strVbox);
4606 }
4607
4608#ifdef VBOX_WITH_AHCI
4609// <const name="HardDiskControllerSATA" value="7" />
4610 rc = GetStorageControllerByName(Bstr("SATA"), pController.asOutParam());
4611 if (SUCCEEDED(rc))
4612 {
4613 strVbox = "AHCI";
4614 lSATAControllerIndex = (int32_t)pNewDesc->m->llDescriptions.size();
4615 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerSATA,
4616 Utf8StrFmt("%d", lSATAControllerIndex),
4617 strVbox,
4618 strVbox);
4619 }
4620#endif // VBOX_WITH_AHCI
4621
4622#ifdef VBOX_WITH_LSILOGIC
4623// <const name="HardDiskControllerSCSI" value="8" />
4624 rc = GetStorageControllerByName(Bstr("SCSI"), pController.asOutParam());
4625 if (SUCCEEDED(rc))
4626 {
4627 rc = pController->COMGETTER(ControllerType)(&ctlr);
4628 if (SUCCEEDED(rc))
4629 {
4630 strVbox = "LsiLogic"; // the default in VBox
4631 switch(ctlr)
4632 {
4633 case StorageControllerType_LsiLogic: strVbox = "LsiLogic"; break;
4634 case StorageControllerType_BusLogic: strVbox = "BusLogic"; break;
4635 }
4636 lSCSIControllerIndex = (int32_t)pNewDesc->m->llDescriptions.size();
4637 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerSCSI,
4638 Utf8StrFmt("%d", lSCSIControllerIndex),
4639 strVbox,
4640 strVbox);
4641 }
4642 else
4643 throw rc;
4644 }
4645#endif // VBOX_WITH_LSILOGIC
4646
4647// <const name="HardDiskImage" value="9" />
4648 HDData::AttachmentList::iterator itA;
4649 for (itA = mHDData->mAttachments.begin();
4650 itA != mHDData->mAttachments.end();
4651 ++itA)
4652 {
4653 ComObjPtr<HardDiskAttachment> pHDA = *itA;
4654
4655 // the attachment's data
4656 ComPtr<IHardDisk> pHardDisk;
4657 ComPtr<IStorageController> ctl;
4658 Bstr controllerName;
4659
4660 rc = pHDA->COMGETTER(Controller)(controllerName.asOutParam());
4661 if (FAILED(rc)) throw rc;
4662
4663 rc = GetStorageControllerByName(controllerName, ctl.asOutParam());
4664 if (FAILED(rc)) throw rc;
4665
4666 StorageBus_T storageBus;
4667 LONG lChannel;
4668 LONG lDevice;
4669
4670 rc = ctl->COMGETTER(Bus)(&storageBus);
4671 if (FAILED(rc)) throw rc;
4672
4673 rc = pHDA->COMGETTER(HardDisk)(pHardDisk.asOutParam());
4674 if (FAILED(rc)) throw rc;
4675
4676 rc = pHDA->COMGETTER(Port)(&lChannel);
4677 if (FAILED(rc)) throw rc;
4678
4679 rc = pHDA->COMGETTER(Device)(&lDevice);
4680 if (FAILED(rc)) throw rc;
4681
4682 Bstr bstrLocation;
4683 rc = pHardDisk->COMGETTER(Location)(bstrLocation.asOutParam());
4684 if (FAILED(rc)) throw rc;
4685 Bstr bstrName;
4686 rc = pHardDisk->COMGETTER(Name)(bstrName.asOutParam());
4687 if (FAILED(rc)) throw rc;
4688
4689 // force reading state, or else size will be returned as 0
4690 MediaState_T ms;
4691 rc = pHardDisk->COMGETTER(State)(&ms);
4692 if (FAILED(rc)) throw rc;
4693
4694 ULONG64 ullSize;
4695 rc = pHardDisk->COMGETTER(Size)(&ullSize);
4696 if (FAILED(rc)) throw rc;
4697
4698 // and how this translates to the virtual system
4699 int32_t lControllerVsys = 0;
4700 LONG lChannelVsys;
4701
4702 switch (storageBus)
4703 {
4704 case StorageBus_IDE:
4705 // this is the exact reverse to what we're doing in Appliance::taskThreadImportMachines,
4706 // and it must be updated when that is changed!
4707
4708 if (lChannel == 0 && lDevice == 0) // primary master
4709 lChannelVsys = 0;
4710 else if (lChannel == 0 && lDevice == 1) // primary slave
4711 lChannelVsys = 1;
4712 else if (lChannel == 1 && lDevice == 1) // secondary slave; secondary master is always CDROM
4713 lChannelVsys = 2;
4714 else
4715 throw setError(VBOX_E_NOT_SUPPORTED,
4716 tr("Cannot handle hard disk attachment: channel is %d, device is %d"), lChannel, lDevice);
4717
4718 lControllerVsys = lIDEControllerIndex;
4719 break;
4720
4721 case StorageBus_SATA:
4722 lChannelVsys = lChannel; // should be between 0 and 29
4723 lControllerVsys = lSATAControllerIndex;
4724 break;
4725
4726 case StorageBus_SCSI:
4727 lChannelVsys = lChannel; // should be between 0 and 15
4728 lControllerVsys = lSCSIControllerIndex;
4729 break;
4730
4731 default:
4732 throw setError(VBOX_E_NOT_SUPPORTED,
4733 tr("Cannot handle hard disk attachment: storageBus is %d, channel is %d, device is %d"), storageBus, lChannel, lDevice);
4734 break;
4735 }
4736
4737 Utf8Str strTargetVmdkName(bstrName);
4738 RTPathStripExt(strTargetVmdkName.mutableRaw());
4739 strTargetVmdkName.append(".vmdk");
4740
4741 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskImage,
4742 strTargetVmdkName, // disk ID: let's use the name
4743 strTargetVmdkName, // OVF value:
4744 Utf8Str(bstrLocation), // vbox value: media path
4745 (uint32_t)(ullSize / _1M),
4746 Utf8StrFmt("controller=%RI32;channel=%RI32", lControllerVsys, lChannelVsys));
4747 }
4748
4749 /* Floppy Drive */
4750 if (fFloppyEnabled)
4751 pNewDesc->addEntry(VirtualSystemDescriptionType_Floppy, "", "", "");
4752
4753 /* CD Drive */
4754 if (fDVDEnabled)
4755 pNewDesc->addEntry(VirtualSystemDescriptionType_CDROM, "", "", "");
4756
4757// <const name="NetworkAdapter" />
4758 size_t a;
4759 for (a = 0;
4760 a < SchemaDefs::NetworkAdapterCount;
4761 ++a)
4762 {
4763 ComPtr<INetworkAdapter> pNetworkAdapter;
4764 BOOL fEnabled;
4765 NetworkAdapterType_T adapterType;
4766 NetworkAttachmentType_T attachmentType;
4767
4768 rc = GetNetworkAdapter((ULONG)a, pNetworkAdapter.asOutParam());
4769 if (FAILED(rc)) throw rc;
4770 /* Enable the network card & set the adapter type */
4771 rc = pNetworkAdapter->COMGETTER(Enabled)(&fEnabled);
4772 if (FAILED(rc)) throw rc;
4773
4774 if (fEnabled)
4775 {
4776 Utf8Str strAttachmentType;
4777
4778 rc = pNetworkAdapter->COMGETTER(AdapterType)(&adapterType);
4779 if (FAILED(rc)) throw rc;
4780
4781 rc = pNetworkAdapter->COMGETTER(AttachmentType)(&attachmentType);
4782 if (FAILED(rc)) throw rc;
4783
4784 switch (attachmentType)
4785 {
4786 case NetworkAttachmentType_Null:
4787 strAttachmentType = "Null";
4788 break;
4789
4790 case NetworkAttachmentType_NAT:
4791 strAttachmentType = "NAT";
4792 break;
4793
4794 case NetworkAttachmentType_Bridged:
4795 strAttachmentType = "Bridged";
4796 break;
4797
4798 case NetworkAttachmentType_Internal:
4799 strAttachmentType = "Internal";
4800 break;
4801
4802 case NetworkAttachmentType_HostOnly:
4803 strAttachmentType = "HostOnly";
4804 break;
4805 }
4806
4807 pNewDesc->addEntry(VirtualSystemDescriptionType_NetworkAdapter,
4808 "", // ref
4809 strAttachmentType, // orig
4810 Utf8StrFmt("%RI32", (uint32_t)adapterType), // conf
4811 0,
4812 Utf8StrFmt("type=%s", strAttachmentType.c_str())); // extra conf
4813 }
4814 }
4815
4816// <const name="USBController" />
4817#ifdef VBOX_WITH_USB
4818 if (fUSBEnabled)
4819 pNewDesc->addEntry(VirtualSystemDescriptionType_USBController, "", "", "");
4820#endif /* VBOX_WITH_USB */
4821
4822// <const name="SoundCard" />
4823 if (fAudioEnabled)
4824 {
4825 pNewDesc->addEntry(VirtualSystemDescriptionType_SoundCard,
4826 "",
4827 "ensoniq1371", // this is what OVFTool writes and VMware supports
4828 Utf8StrFmt("%RI32", audioController));
4829 }
4830
4831 // finally, add the virtual system to the appliance
4832 Appliance *pAppliance = static_cast<Appliance*>(aAppliance);
4833 AutoCaller autoCaller1(pAppliance);
4834 CheckComRCReturnRC(autoCaller1.rc());
4835
4836 /* We return the new description to the caller */
4837 ComPtr<IVirtualSystemDescription> copy(pNewDesc);
4838 copy.queryInterfaceTo(aDescription);
4839
4840 AutoWriteLock alock(pAppliance);
4841
4842 pAppliance->m->virtualSystemDescriptions.push_back(pNewDesc);
4843 }
4844 catch(HRESULT arc)
4845 {
4846 rc = arc;
4847 }
4848
4849 return rc;
4850}
4851
4852/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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