VirtualBox

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

Last change on this file since 18028 was 18024, checked in by vboxsync, 16 years ago

OVF: add GetDescriptionByType COM call

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 168.7 KB
Line 
1/* $Id: ApplianceImpl.cpp 18024 2009-03-17 14:00:08Z 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
28#include "ApplianceImpl.h"
29#include "VirtualBoxImpl.h"
30#include "GuestOSTypeImpl.h"
31#include "ProgressImpl.h"
32#include "MachineImpl.h"
33#include "HostNetworkInterfaceImpl.h"
34
35#include "Logging.h"
36
37#include "VBox/xml.h"
38
39using namespace std;
40
41////////////////////////////////////////////////////////////////////////////////
42//
43// hardware definitions
44//
45////////////////////////////////////////////////////////////////////////////////
46
47struct DiskImage
48{
49 Utf8Str strDiskId; // value from DiskSection/Disk/@diskId
50 int64_t iCapacity; // value from DiskSection/Disk/@capacity;
51 // (maximum size for dynamic images, I guess; we always translate this to bytes)
52 int64_t iPopulatedSize; // value from DiskSection/Disk/@populatedSize
53 // (actual used size of disk, always in bytes; can be an estimate of used disk
54 // space, but cannot be larger than iCapacity)
55 Utf8Str strFormat; // value from DiskSection/Disk/@format
56 // typically http://www.vmware.com/specifications/vmdk.html#sparse
57
58 // fields from /References/File; the spec says the file reference from disk can be empty,
59 // so in that case, strFilename will be empty, then a new disk should be created
60 Utf8Str strHref; // value from /References/File/@href (filename); if empty, then the remaining fields are ignored
61 int64_t iSize; // value from /References/File/@size (optional according to spec; then we set -1 here)
62 int64_t iChunkSize; // value from /References/File/@chunkSize (optional, unsupported)
63 Utf8Str strCompression; // value from /References/File/@compression (optional, can be "gzip" according to spec)
64};
65
66struct VirtualHardwareItem
67{
68 Utf8Str strDescription;
69 Utf8Str strCaption;
70 Utf8Str strElementName;
71
72 uint32_t ulInstanceID;
73 uint32_t ulParent;
74
75 OVFResourceType_T resourceType;
76 Utf8Str strOtherResourceType;
77 Utf8Str strResourceSubType;
78
79 Utf8Str strHostResource; // "Abstractly specifies how a device shall connect to a resource on the deployment platform.
80 // Not all devices need a backing." Used with disk items, for which this references a virtual
81 // disk from the Disks section.
82 bool fAutomaticAllocation;
83 bool fAutomaticDeallocation;
84 Utf8Str strConnection; // "All Ethernet adapters that specify the same abstract network connection name within an OVF
85 // package shall be deployed on the same network. The abstract network connection name shall be
86 // listed in the NetworkSection at the outermost envelope level." We ignore this and only set up
87 // a network adapter depending on the network name.
88 Utf8Str strAddress; // "Device-specific. For an Ethernet adapter, this specifies the MAC address."
89 Utf8Str strAddressOnParent; // "For a device, this specifies its location on the controller."
90 Utf8Str strAllocationUnits; // "Specifies the units of allocation used. For example, “byte * 2^20”."
91 uint64_t ullVirtualQuantity; // "Specifies the quantity of resources presented. For example, “256”."
92 uint64_t ullReservation; // "Specifies the minimum quantity of resources guaranteed to be available."
93 uint64_t ullLimit; // "Specifies the maximum quantity of resources that will be granted."
94 uint64_t ullWeight; // "Specifies a relative priority for this allocation in relation to other allocations."
95
96 Utf8Str strConsumerVisibility;
97 Utf8Str strMappingBehavior;
98 Utf8Str strPoolID;
99 uint32_t ulBusNumber; // seen with IDE controllers, but not listed in OVF spec
100
101 uint32_t ulLineNumber; // line number of <Item> element in XML source; cached for error messages
102
103 VirtualHardwareItem()
104 : ulInstanceID(0), fAutomaticAllocation(false), fAutomaticDeallocation(false), ullVirtualQuantity(0), ullReservation(0), ullLimit(0), ullWeight(0), ulBusNumber(0), ulLineNumber(0)
105 {};
106};
107
108typedef map<Utf8Str, DiskImage> DiskImagesMap;
109
110struct VirtualSystem;
111
112typedef map<uint32_t, VirtualHardwareItem> HardwareItemsMap;
113
114struct HardDiskController
115{
116 uint32_t idController; // instance ID (Item/InstanceId); this gets referenced from HardDisk
117 enum ControllerSystemType { IDE, SATA, SCSI };
118 ControllerSystemType system; // one of IDE, SATA, SCSI
119 Utf8Str strControllerType; // controller subtype (Item/ResourceSubType); e.g. "LsiLogic"; can be empty (esp. for IDE)
120 Utf8Str strAddress; // for IDE
121 uint32_t ulBusNumber; // for IDE
122
123 HardDiskController()
124 : idController(0),
125 ulBusNumber(0)
126 {
127 }
128};
129
130typedef map<uint32_t, HardDiskController> ControllersMap;
131
132struct VirtualDisk
133{
134 uint32_t idController; // SCSI (or IDE) controller this disk is connected to;
135 // points into VirtualSystem.mapControllers
136 uint32_t ulAddressOnParent; // parsed strAddressOnParent of hardware item; will be 0 or 1 for IDE
137 // and possibly higher for disks attached to SCSI controllers (untested)
138 Utf8Str strDiskId; // if the hard disk has an ovf:/disk/<id> reference,
139 // this receives the <id> component; points to one of the
140 // references in Appliance::Data.mapDisks
141};
142
143typedef map<Utf8Str, VirtualDisk> VirtualDisksMap;
144
145struct EthernetAdapter
146{
147 Utf8Str strAdapterType; // "PCNet32" or "E1000" or whatever; from <rasd:ResourceSubType>
148 Utf8Str strNetworkName; // from <rasd:Connection>
149};
150
151typedef list<EthernetAdapter> EthernetAdaptersList;
152
153struct VirtualSystem
154{
155 Utf8Str strName; // copy of VirtualSystem/@id
156
157 Utf8Str strDescription; // copy of VirtualSystem/Info content
158
159 CIMOSType_T cimos;
160 Utf8Str strVirtualSystemType; // generic hardware description; OVF says this can be something like "vmx-4" or "xen";
161 // VMware Workstation 6.5 is "vmx-07"
162
163 HardwareItemsMap mapHardwareItems; // map of virtual hardware items, sorted by unique instance ID
164
165 uint64_t ullMemorySize; // always in bytes, copied from llHardwareItems; default = 0 (unspecified)
166 uint16_t cCPUs; // no. of CPUs, copied from llHardwareItems; default = 1
167
168 EthernetAdaptersList llEthernetAdapters; // (one for each VirtualSystem/Item[@ResourceType=10]element)
169
170 ControllersMap mapControllers;
171 // list of hard disk controllers
172 // (one for each VirtualSystem/Item[@ResourceType=6] element with accumulated data from children)
173
174 VirtualDisksMap mapVirtualDisks;
175 // (one for each VirtualSystem/Item[@ResourceType=17] element with accumulated data from children)
176
177 bool fHasFloppyDrive; // true if there's a floppy item in mapHardwareItems
178 bool fHasCdromDrive; // true if there's a CD-ROM item in mapHardwareItems; ISO images are not yet supported by OVFtool
179 bool fHasUsbController; // true if there's a USB controller item in mapHardwareItems
180
181 Utf8Str strSoundCardType; // if not empty, then the system wants a soundcard; this then specifies the hardware;
182 // VMware Workstation 6.5 uses "ensoniq1371" for example
183
184 Utf8Str strLicenceInfo; // license info if any; receives contents of VirtualSystem/EulaSection/Info
185 Utf8Str strLicenceText; // license info if any; receives contents of VirtualSystem/EulaSection/License
186
187 VirtualSystem()
188 : ullMemorySize(0), cCPUs(1), fHasFloppyDrive(false), fHasCdromDrive(false), fHasUsbController(false)
189 {
190 }
191};
192
193////////////////////////////////////////////////////////////////////////////////
194//
195// Appliance data definition
196//
197////////////////////////////////////////////////////////////////////////////////
198
199// opaque private instance data of Appliance class
200struct Appliance::Data
201{
202 Utf8Str strPath; // file name last given to either read() or write()
203
204 DiskImagesMap mapDisks; // map of DiskImage structs, sorted by DiskImage.strDiskId
205
206 list<VirtualSystem> llVirtualSystems; // list of virtual systems, created by and valid after read()
207
208 list< ComObjPtr<VirtualSystemDescription> > virtualSystemDescriptions; //
209
210 list<Utf8Str> llWarnings;
211};
212
213struct VirtualSystemDescription::Data
214{
215 list<VirtualSystemDescriptionEntry> llDescriptions;
216};
217
218////////////////////////////////////////////////////////////////////////////////
219//
220// Threads
221//
222////////////////////////////////////////////////////////////////////////////////
223
224struct Appliance::TaskImportMachines
225{
226 TaskImportMachines(Appliance *aThat, Progress *aProgress)
227 : pAppliance(aThat)
228 , progress(aProgress)
229 , rc(S_OK)
230 {}
231 ~TaskImportMachines() {}
232
233 HRESULT startThread();
234
235 Appliance *pAppliance;
236 ComObjPtr<Progress> progress;
237 HRESULT rc;
238};
239
240struct Appliance::TaskWriteOVF
241{
242 TaskWriteOVF(Appliance *aThat, Progress *aProgress)
243 : pAppliance(aThat)
244 , progress(aProgress)
245 , rc(S_OK)
246 {}
247 ~TaskWriteOVF() {}
248
249 HRESULT startThread();
250
251 Appliance *pAppliance;
252 ComObjPtr<Progress> progress;
253 HRESULT rc;
254};
255
256////////////////////////////////////////////////////////////////////////////////
257//
258// internal helpers
259//
260////////////////////////////////////////////////////////////////////////////////
261
262static Utf8Str stripFilename(const Utf8Str &strFile)
263{
264 Utf8Str str2(strFile);
265 RTPathStripFilename(str2.mutableRaw());
266 return str2;
267}
268
269static const struct
270{
271 CIMOSType_T cim;
272 const char *pcszVbox;
273}
274 g_osTypes[] =
275 {
276 { CIMOSType_CIMOS_Unknown, SchemaDefs_OSTypeId_Other },
277 { CIMOSType_CIMOS_OS2, SchemaDefs_OSTypeId_OS2 },
278 { CIMOSType_CIMOS_MSDOS, SchemaDefs_OSTypeId_DOS },
279 { CIMOSType_CIMOS_WIN3x, SchemaDefs_OSTypeId_Windows31 },
280 { CIMOSType_CIMOS_WIN95, SchemaDefs_OSTypeId_Windows95 },
281 { CIMOSType_CIMOS_WIN98, SchemaDefs_OSTypeId_Windows98 },
282 { CIMOSType_CIMOS_WINNT, SchemaDefs_OSTypeId_WindowsNT4 },
283 { CIMOSType_CIMOS_NetWare, SchemaDefs_OSTypeId_Netware },
284 { CIMOSType_CIMOS_NovellOES, SchemaDefs_OSTypeId_Netware },
285 { CIMOSType_CIMOS_Solaris, SchemaDefs_OSTypeId_OpenSolaris },
286 { CIMOSType_CIMOS_SunOS, SchemaDefs_OSTypeId_OpenSolaris },
287 { CIMOSType_CIMOS_FreeBSD, SchemaDefs_OSTypeId_FreeBSD },
288 { CIMOSType_CIMOS_NetBSD, SchemaDefs_OSTypeId_NetBSD },
289 { CIMOSType_CIMOS_QNX, SchemaDefs_OSTypeId_QNX },
290 { CIMOSType_CIMOS_Windows2000, SchemaDefs_OSTypeId_Windows2000 },
291 { CIMOSType_CIMOS_WindowsMe, SchemaDefs_OSTypeId_WindowsMe },
292 { CIMOSType_CIMOS_OpenBSD, SchemaDefs_OSTypeId_OpenBSD },
293 { CIMOSType_CIMOS_WindowsXP, SchemaDefs_OSTypeId_WindowsXP },
294 { CIMOSType_CIMOS_WindowsXPEmbedded, SchemaDefs_OSTypeId_WindowsXP },
295 { CIMOSType_CIMOS_WindowsEmbeddedforPointofService, SchemaDefs_OSTypeId_WindowsXP },
296 { CIMOSType_CIMOS_MicrosoftWindowsServer2003, SchemaDefs_OSTypeId_Windows2003 },
297 { CIMOSType_CIMOS_MicrosoftWindowsServer2003_64, SchemaDefs_OSTypeId_Windows2003_64 },
298 { CIMOSType_CIMOS_WindowsXP_64, SchemaDefs_OSTypeId_WindowsXP_64 },
299 { CIMOSType_CIMOS_WindowsVista, SchemaDefs_OSTypeId_WindowsVista },
300 { CIMOSType_CIMOS_WindowsVista_64, SchemaDefs_OSTypeId_WindowsVista_64 },
301 { CIMOSType_CIMOS_MicrosoftWindowsServer2008, SchemaDefs_OSTypeId_Windows2008 },
302 { CIMOSType_CIMOS_MicrosoftWindowsServer2008_64, SchemaDefs_OSTypeId_Windows2008_64 },
303 { CIMOSType_CIMOS_FreeBSD_64, SchemaDefs_OSTypeId_FreeBSD_64 },
304 { CIMOSType_CIMOS_RedHatEnterpriseLinux, SchemaDefs_OSTypeId_RedHat },
305 { CIMOSType_CIMOS_RedHatEnterpriseLinux_64, SchemaDefs_OSTypeId_RedHat_64 },
306 { CIMOSType_CIMOS_Solaris_64, SchemaDefs_OSTypeId_OpenSolaris_64 },
307 { CIMOSType_CIMOS_SUSE, SchemaDefs_OSTypeId_OpenSUSE },
308 { CIMOSType_CIMOS_SLES, SchemaDefs_OSTypeId_OpenSUSE },
309 { CIMOSType_CIMOS_NovellLinuxDesktop, SchemaDefs_OSTypeId_OpenSUSE },
310 { CIMOSType_CIMOS_SUSE_64, SchemaDefs_OSTypeId_OpenSUSE_64 },
311 { CIMOSType_CIMOS_SLES_64, SchemaDefs_OSTypeId_OpenSUSE_64 },
312 { CIMOSType_CIMOS_LINUX, SchemaDefs_OSTypeId_Linux },
313 { CIMOSType_CIMOS_SunJavaDesktopSystem, SchemaDefs_OSTypeId_Linux },
314 { CIMOSType_CIMOS_TurboLinux, SchemaDefs_OSTypeId_Linux},
315
316 // { CIMOSType_CIMOS_TurboLinux_64, },
317 // { CIMOSType_CIMOS_Linux_64, },
318 // osTypeVBox = VBOXOSTYPE_Linux_x64;
319 // break;
320
321 { CIMOSType_CIMOS_Mandriva, SchemaDefs_OSTypeId_Mandriva },
322 { CIMOSType_CIMOS_Mandriva_64, SchemaDefs_OSTypeId_Mandriva_64 },
323 { CIMOSType_CIMOS_Ubuntu, SchemaDefs_OSTypeId_Ubuntu },
324 { CIMOSType_CIMOS_Ubuntu_64, SchemaDefs_OSTypeId_Ubuntu_64 },
325 { CIMOSType_CIMOS_Debian, SchemaDefs_OSTypeId_Debian },
326 { CIMOSType_CIMOS_Debian_64, SchemaDefs_OSTypeId_Debian_64 },
327 { CIMOSType_CIMOS_Linux_2_4_x, SchemaDefs_OSTypeId_Linux24 },
328 { CIMOSType_CIMOS_Linux_2_4_x_64, SchemaDefs_OSTypeId_Linux24_64 },
329 { CIMOSType_CIMOS_Linux_2_6_x, SchemaDefs_OSTypeId_Linux26 },
330 { CIMOSType_CIMOS_Linux_2_6_x_64, SchemaDefs_OSTypeId_Linux26_64 }
331};
332
333/**
334 * Private helper func that suggests a VirtualBox guest OS type
335 * for the given OVF operating system type.
336 * @param osTypeVBox
337 * @param c
338 */
339static void convertCIMOSType2VBoxOSType(Utf8Str &strType, CIMOSType_T c)
340{
341 for (size_t i = 0; i < RT_ELEMENTS(g_osTypes); ++i)
342 {
343 if (c == g_osTypes[i].cim)
344 {
345 strType = g_osTypes[i].pcszVbox;
346 return;
347 }
348 }
349
350 strType = SchemaDefs_OSTypeId_Other;
351}
352
353/**
354 * Private helper func that suggests a VirtualBox guest OS type
355 * for the given OVF operating system type.
356 * @param osTypeVBox
357 * @param c
358 */
359static CIMOSType_T convertVBoxOSType2CIMOSType(const char *pcszVbox)
360{
361 for (size_t i = 0; i < RT_ELEMENTS(g_osTypes); ++i)
362 {
363 if (!RTStrICmp(pcszVbox, g_osTypes[i].pcszVbox))
364 return g_osTypes[i].cim;
365 }
366
367 return CIMOSType_CIMOS_Other;
368}
369
370////////////////////////////////////////////////////////////////////////////////
371//
372// IVirtualBox public methods
373//
374////////////////////////////////////////////////////////////////////////////////
375
376// This code is here so we won't have to include the appliance headers in the
377// IVirtualBox implementation.
378
379/**
380 * Implementation for IVirtualBox::createAppliance.
381 *
382 * @param anAppliance IAppliance object created if S_OK is returned.
383 * @return S_OK or error.
384 */
385STDMETHODIMP VirtualBox::CreateAppliance(IAppliance** anAppliance)
386{
387 HRESULT rc;
388
389 ComObjPtr<Appliance> appliance;
390 appliance.createObject();
391 rc = appliance->init(this);
392
393 if (SUCCEEDED(rc))
394 appliance.queryInterfaceTo(anAppliance);
395
396 return rc;
397}
398
399////////////////////////////////////////////////////////////////////////////////
400//
401// Appliance::task methods
402//
403////////////////////////////////////////////////////////////////////////////////
404
405HRESULT Appliance::TaskImportMachines::startThread()
406{
407 int vrc = RTThreadCreate(NULL, Appliance::taskThreadImportMachines, this,
408 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
409 "Appliance::Task");
410 ComAssertMsgRCRet(vrc,
411 ("Could not create taskThreadImportMachines (%Rrc)\n", vrc), E_FAIL);
412
413 return S_OK;
414}
415
416HRESULT Appliance::TaskWriteOVF::startThread()
417{
418 int vrc = RTThreadCreate(NULL, Appliance::taskThreadWriteOVF, this,
419 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0,
420 "Appliance::Task");
421 ComAssertMsgRCRet(vrc,
422 ("Could not create taskThreadExportOVF (%Rrc)\n", vrc), E_FAIL);
423
424 return S_OK;
425}
426
427////////////////////////////////////////////////////////////////////////////////
428//
429// Appliance constructor / destructor
430//
431////////////////////////////////////////////////////////////////////////////////
432
433DEFINE_EMPTY_CTOR_DTOR(Appliance)
434struct shutup {};
435
436/**
437 * Appliance COM initializer.
438 * @param
439 * @return
440 */
441
442HRESULT Appliance::init(VirtualBox *aVirtualBox)
443{
444 /* Enclose the state transition NotReady->InInit->Ready */
445 AutoInitSpan autoInitSpan(this);
446 AssertReturn(autoInitSpan.isOk(), E_FAIL);
447
448 /* Weak reference to a VirtualBox object */
449 unconst(mVirtualBox) = aVirtualBox;
450
451 // initialize data
452 m = new Data;
453
454 /* Confirm a successful initialization */
455 autoInitSpan.setSucceeded();
456
457 return S_OK;
458}
459
460/**
461 * Appliance COM uninitializer.
462 * @return
463 */
464void Appliance::uninit()
465{
466 delete m;
467 m = NULL;
468}
469
470////////////////////////////////////////////////////////////////////////////////
471//
472// Appliance private methods
473//
474////////////////////////////////////////////////////////////////////////////////
475
476/**
477 * Private helper method that goes thru the elements of the given "current" element in the OVF XML
478 * and handles the contained child elements (which can be "Section" or "Content" elements).
479 *
480 * @param pcszPath Path spec of the XML file, for error messages.
481 * @param pReferencesElement "References" element from OVF, for looking up file specifications; can be NULL if no such element is present.
482 * @param pCurElem Element whose children are to be analyzed here.
483 * @return
484 */
485HRESULT Appliance::LoopThruSections(const char *pcszPath,
486 const xml::ElementNode *pReferencesElem,
487 const xml::ElementNode *pCurElem)
488{
489 HRESULT rc;
490
491 xml::NodesLoop loopChildren(*pCurElem);
492 const xml::ElementNode *pElem;
493 while ((pElem = loopChildren.forAllNodes()))
494 {
495 const char *pcszElemName = pElem->getName();
496 const char *pcszTypeAttr = "";
497 const xml::AttributeNode *pTypeAttr;
498 if ((pTypeAttr = pElem->findAttribute("type")))
499 pcszTypeAttr = pTypeAttr->getValue();
500
501 if ( (!strcmp(pcszElemName, "DiskSection"))
502 || ( (!strcmp(pcszElemName, "Section"))
503 && (!strcmp(pcszTypeAttr, "ovf:DiskSection_Type"))
504 )
505 )
506 {
507 if (!(SUCCEEDED((rc = HandleDiskSection(pcszPath, pReferencesElem, pElem)))))
508 return rc;
509 }
510 else if ( (!strcmp(pcszElemName, "NetworkSection")) // we ignore NetworkSections for now
511 || ( (!strcmp(pcszElemName, "Section"))
512 && (!strcmp(pcszTypeAttr, "ovf:NetworkSection_Type"))
513 )
514 )
515 {
516 if (!(SUCCEEDED((rc = HandleNetworkSection(pcszPath, pElem)))))
517 return rc;
518 }
519 else if ( (!strcmp(pcszElemName, "DeploymentOptionSection>")))
520 {
521 // TODO
522 }
523 else if ( (!strcmp(pcszElemName, "Info")))
524 {
525 // child of VirtualSystemCollection -- TODO
526 }
527 else if ( (!strcmp(pcszElemName, "ResourceAllocationSection")))
528 {
529 // child of VirtualSystemCollection -- TODO
530 }
531 else if ( (!strcmp(pcszElemName, "StartupSection")))
532 {
533 // child of VirtualSystemCollection -- TODO
534 }
535 else if ( (!strcmp(pcszElemName, "VirtualSystem"))
536 || ( (!strcmp(pcszElemName, "Content"))
537 && (!strcmp(pcszTypeAttr, "ovf:VirtualSystem_Type"))
538 )
539 )
540 {
541 if (!(SUCCEEDED((rc = HandleVirtualSystemContent(pcszPath, pElem)))))
542 return rc;
543 }
544 else if ( (!strcmp(pcszElemName, "VirtualSystemCollection"))
545 || ( (!strcmp(pcszElemName, "Content"))
546 && (!strcmp(pcszTypeAttr, "ovf:VirtualSystemCollection_Type"))
547 )
548 )
549 {
550 // TODO ResourceAllocationSection
551
552 // recurse for this, since it has VirtualSystem elements as children
553 if (!(SUCCEEDED((rc = LoopThruSections(pcszPath, pReferencesElem, pElem)))))
554 return rc;
555 }
556 }
557
558 return S_OK;
559}
560
561/**
562 * Private helper method that handles disk sections in the OVF XML.
563 * Gets called indirectly from IAppliance::read().
564 *
565 * @param pcszPath Path spec of the XML file, for error messages.
566 * @param pReferencesElement "References" element from OVF, for looking up file specifications; can be NULL if no such element is present.
567 * @param pSectionElem Section element for which this helper is getting called.
568 * @return
569 */
570HRESULT Appliance::HandleDiskSection(const char *pcszPath,
571 const xml::ElementNode *pReferencesElem,
572 const xml::ElementNode *pSectionElem)
573{
574 // contains "Disk" child elements
575 xml::NodesLoop loopDisks(*pSectionElem, "Disk");
576 const xml::ElementNode *pelmDisk;
577 while ((pelmDisk = loopDisks.forAllNodes()))
578 {
579 DiskImage d;
580 const char *pcszBad = NULL;
581 if (!(pelmDisk->getAttributeValue("diskId", d.strDiskId)))
582 pcszBad = "diskId";
583 else if (!(pelmDisk->getAttributeValue("format", d.strFormat)))
584 pcszBad = "format";
585 else if (!(pelmDisk->getAttributeValue("capacity", d.iCapacity)))
586 pcszBad = "capacity";
587 else
588 {
589 if (!(pelmDisk->getAttributeValue("populatedSize", d.iPopulatedSize)))
590 // optional
591 d.iPopulatedSize = -1;
592
593 Utf8Str strFileRef;
594 if (pelmDisk->getAttributeValue("fileRef", strFileRef)) // optional
595 {
596 // look up corresponding /References/File nodes (list built above)
597 const xml::ElementNode *pFileElem;
598 if ( pReferencesElem
599 && ((pFileElem = pReferencesElem->findChildElementFromId(strFileRef.c_str())))
600 )
601 {
602 // copy remaining values from file node then
603 const char *pcszBadInFile = NULL;
604 if (!(pFileElem->getAttributeValue("href", d.strHref)))
605 pcszBadInFile = "href";
606 else if (!(pFileElem->getAttributeValue("size", d.iSize)))
607 d.iSize = -1; // optional
608 // if (!(pFileElem->getAttributeValue("size", d.iChunkSize))) TODO
609 d.iChunkSize = -1; // optional
610 pFileElem->getAttributeValue("compression", d.strCompression);
611
612 if (pcszBadInFile)
613 return setError(VBOX_E_FILE_ERROR,
614 tr("Error reading \"%s\": missing or invalid attribute '%s' in 'File' element, line %d"),
615 pcszPath,
616 pcszBadInFile,
617 pFileElem->getLineNumber());
618 }
619 else
620 return setError(VBOX_E_FILE_ERROR,
621 tr("Error reading \"%s\": cannot find References/File element for ID '%s' referenced by 'Disk' element, line %d"),
622 pcszPath,
623 strFileRef.c_str(),
624 pelmDisk->getLineNumber());
625 }
626 }
627
628 if (pcszBad)
629 return setError(VBOX_E_FILE_ERROR,
630 tr("Error reading \"%s\": missing or invalid attribute '%s' in 'DiskSection' element, line %d"),
631 pcszPath,
632 pcszBad,
633 pelmDisk->getLineNumber());
634
635 m->mapDisks[d.strDiskId] = d;
636 }
637
638 return S_OK;
639}
640
641/**
642 * Private helper method that handles network sections in the OVF XML.
643 * Gets called indirectly from IAppliance::read().
644 *
645 * @param pcszPath Path spec of the XML file, for error messages.
646 * @param pSectionElem Section element for which this helper is getting called.
647 * @return
648 */
649HRESULT Appliance::HandleNetworkSection(const char * /* pcszPath */,
650 const xml::ElementNode * /* pSectionElem */)
651{
652 // we ignore network sections for now
653
654// xml::NodesLoop loopNetworks(*pSectionElem, "Network");
655// const xml::Node *pelmNetwork;
656// while ((pelmNetwork = loopNetworks.forAllNodes()))
657// {
658// Network n;
659// if (!(pelmNetwork->getAttributeValue("name", n.strNetworkName)))
660// return setError(VBOX_E_FILE_ERROR,
661// tr("Error reading \"%s\": missing 'name' attribute in 'Network', line %d"),
662// pcszPath,
663// pelmNetwork->getLineNumber());
664//
665// m->mapNetworks[n.strNetworkName] = n;
666// }
667
668 return S_OK;
669}
670
671/**
672 * Private helper method that handles a "VirtualSystem" element in the OVF XML.
673 * Gets called indirectly from IAppliance::read().
674 *
675 * @param pcszPath
676 * @param pContentElem
677 * @return
678 */
679HRESULT Appliance::HandleVirtualSystemContent(const char *pcszPath,
680 const xml::ElementNode *pelmVirtualSystem)
681{
682 VirtualSystem vsys;
683
684 const xml::AttributeNode *pIdAttr = pelmVirtualSystem->findAttribute("id");
685 if (pIdAttr)
686 vsys.strName = pIdAttr->getValue();
687
688 xml::NodesLoop loop(*pelmVirtualSystem); // all child elements
689 const xml::ElementNode *pelmThis;
690 while ((pelmThis = loop.forAllNodes()))
691 {
692 const char *pcszElemName = pelmThis->getName();
693 const xml::AttributeNode *pTypeAttr = pelmThis->findAttribute("type");
694 const char *pcszTypeAttr = (pTypeAttr) ? pTypeAttr->getValue() : "";
695
696 if (!strcmp(pcszElemName, "Info"))
697 {
698 // the "Info" section directly under VirtualSystem is where our export routine
699 // chooses to save the VM description (comments), so re-import it from there too
700 vsys.strDescription = pelmThis->getValue();
701 }
702 else if (!strcmp(pcszElemName, "EulaSection"))
703 {
704 /* <EulaSection>
705 <Info ovf:msgid="6">License agreement for the Virtual System.</Info>
706 <License ovf:msgid="1">License terms can go in here.</License>
707 </EulaSection> */
708
709 const xml::ElementNode *pelmInfo, *pelmLicense;
710 if ( ((pelmInfo = pelmThis->findChildElement("Info")))
711 && ((pelmLicense = pelmThis->findChildElement("License")))
712 )
713 {
714 vsys.strLicenceInfo = pelmInfo->getValue();
715 vsys.strLicenceText = pelmLicense->getValue();
716 }
717 }
718 else if ( (!strcmp(pcszElemName, "VirtualHardwareSection"))
719 || (!strcmp(pcszTypeAttr, "ovf:VirtualHardwareSection_Type"))
720 )
721 {
722 const xml::ElementNode *pelmSystem, *pelmVirtualSystemType;
723 if ((pelmSystem = pelmThis->findChildElement("System")))
724 {
725 /* <System>
726 <vssd:Description>Description of the virtual hardware section.</vssd:Description>
727 <vssd:ElementName>vmware</vssd:ElementName>
728 <vssd:InstanceID>1</vssd:InstanceID>
729 <vssd:VirtualSystemIdentifier>MyLampService</vssd:VirtualSystemIdentifier>
730 <vssd:VirtualSystemType>vmx-4</vssd:VirtualSystemType>
731 </System>*/
732 if ((pelmVirtualSystemType = pelmSystem->findChildElement("VirtualSystemType")))
733 vsys.strVirtualSystemType = pelmVirtualSystemType->getValue();
734 }
735
736 xml::NodesLoop loopVirtualHardwareItems(*pelmThis, "Item"); // all "Item" child elements
737 const xml::ElementNode *pelmItem;
738 while ((pelmItem = loopVirtualHardwareItems.forAllNodes()))
739 {
740 VirtualHardwareItem i;
741
742 i.ulLineNumber = pelmItem->getLineNumber();
743
744 xml::NodesLoop loopItemChildren(*pelmItem); // all child elements
745 const xml::ElementNode *pelmItemChild;
746 while ((pelmItemChild = loopItemChildren.forAllNodes()))
747 {
748 const char *pcszItemChildName = pelmItemChild->getName();
749 if (!strcmp(pcszItemChildName, "Description"))
750 i.strDescription = pelmItemChild->getValue();
751 else if (!strcmp(pcszItemChildName, "Caption"))
752 i.strCaption = pelmItemChild->getValue();
753 else if (!strcmp(pcszItemChildName, "ElementName"))
754 i.strElementName = pelmItemChild->getValue();
755 else if ( (!strcmp(pcszItemChildName, "InstanceID"))
756 || (!strcmp(pcszItemChildName, "InstanceId"))
757 )
758 pelmItemChild->copyValue(i.ulInstanceID);
759 else if (!strcmp(pcszItemChildName, "HostResource"))
760 i.strHostResource = pelmItemChild->getValue();
761 else if (!strcmp(pcszItemChildName, "ResourceType"))
762 {
763 uint32_t ulType;
764 pelmItemChild->copyValue(ulType);
765 i.resourceType = (OVFResourceType_T)ulType;
766 }
767 else if (!strcmp(pcszItemChildName, "OtherResourceType"))
768 i.strOtherResourceType = pelmItemChild->getValue();
769 else if (!strcmp(pcszItemChildName, "ResourceSubType"))
770 i.strResourceSubType = pelmItemChild->getValue();
771 else if (!strcmp(pcszItemChildName, "AutomaticAllocation"))
772 i.fAutomaticAllocation = (!strcmp(pelmItemChild->getValue(), "true")) ? true : false;
773 else if (!strcmp(pcszItemChildName, "AutomaticDeallocation"))
774 i.fAutomaticDeallocation = (!strcmp(pelmItemChild->getValue(), "true")) ? true : false;
775 else if (!strcmp(pcszItemChildName, "Parent"))
776 pelmItemChild->copyValue(i.ulParent);
777 else if (!strcmp(pcszItemChildName, "Connection"))
778 i.strConnection = pelmItemChild->getValue();
779 else if (!strcmp(pcszItemChildName, "Address"))
780 i.strAddress = pelmItemChild->getValue();
781 else if (!strcmp(pcszItemChildName, "AddressOnParent"))
782 i.strAddressOnParent = pelmItemChild->getValue();
783 else if (!strcmp(pcszItemChildName, "AllocationUnits"))
784 i.strAllocationUnits = pelmItemChild->getValue();
785 else if (!strcmp(pcszItemChildName, "VirtualQuantity"))
786 pelmItemChild->copyValue(i.ullVirtualQuantity);
787 else if (!strcmp(pcszItemChildName, "Reservation"))
788 pelmItemChild->copyValue(i.ullReservation);
789 else if (!strcmp(pcszItemChildName, "Limit"))
790 pelmItemChild->copyValue(i.ullLimit);
791 else if (!strcmp(pcszItemChildName, "Weight"))
792 pelmItemChild->copyValue(i.ullWeight);
793 else if (!strcmp(pcszItemChildName, "ConsumerVisibility"))
794 i.strConsumerVisibility = pelmItemChild->getValue();
795 else if (!strcmp(pcszItemChildName, "MappingBehavior"))
796 i.strMappingBehavior = pelmItemChild->getValue();
797 else if (!strcmp(pcszItemChildName, "PoolID"))
798 i.strPoolID = pelmItemChild->getValue();
799 else if (!strcmp(pcszItemChildName, "BusNumber"))
800 pelmItemChild->copyValue(i.ulBusNumber);
801 else
802 return setError(VBOX_E_FILE_ERROR,
803 tr("Error reading \"%s\": unknown element \"%s\" under Item element, line %d"),
804 pcszPath,
805 pcszItemChildName,
806 i.ulLineNumber);
807 }
808
809 // store!
810 vsys.mapHardwareItems[i.ulInstanceID] = i;
811 }
812
813 HardwareItemsMap::const_iterator itH;
814
815 for (itH = vsys.mapHardwareItems.begin();
816 itH != vsys.mapHardwareItems.end();
817 ++itH)
818 {
819 const VirtualHardwareItem &i = itH->second;
820
821 // do some analysis
822 switch (i.resourceType)
823 {
824 case OVFResourceType_Processor: // 3
825 /* <rasd:Caption>1 virtual CPU</rasd:Caption>
826 <rasd:Description>Number of virtual CPUs</rasd:Description>
827 <rasd:ElementName>virtual CPU</rasd:ElementName>
828 <rasd:InstanceID>1</rasd:InstanceID>
829 <rasd:ResourceType>3</rasd:ResourceType>
830 <rasd:VirtualQuantity>1</rasd:VirtualQuantity>*/
831 if (i.ullVirtualQuantity < UINT16_MAX)
832 vsys.cCPUs = (uint16_t)i.ullVirtualQuantity;
833 else
834 return setError(VBOX_E_FILE_ERROR,
835 tr("Error reading \"%s\": CPU count %RI64 is larger than %d, line %d"),
836 pcszPath,
837 i.ullVirtualQuantity,
838 UINT16_MAX,
839 i.ulLineNumber);
840 break;
841
842 case OVFResourceType_Memory: // 4
843 if ( (i.strAllocationUnits == "MegaBytes") // found in OVF created by OVF toolkit
844 || (i.strAllocationUnits == "MB") // found in MS docs
845 || (i.strAllocationUnits == "byte * 2^20") // suggested by OVF spec DSP0243 page 21
846 )
847 vsys.ullMemorySize = i.ullVirtualQuantity * 1024 * 1024;
848 else
849 return setError(VBOX_E_FILE_ERROR,
850 tr("Error reading \"%s\": Invalid allocation unit \"%s\" specified with memory size item, line %d"),
851 pcszPath,
852 i.strAllocationUnits.c_str(),
853 i.ulLineNumber);
854 break;
855
856 case OVFResourceType_IDEController: // 5
857 {
858 /* <Item>
859 <rasd:Caption>ideController0</rasd:Caption>
860 <rasd:Description>IDE Controller</rasd:Description>
861 <rasd:InstanceId>5</rasd:InstanceId>
862 <rasd:ResourceType>5</rasd:ResourceType>
863 <rasd:Address>0</rasd:Address>
864 <rasd:BusNumber>0</rasd:BusNumber>
865 </Item> */
866 HardDiskController hdc;
867 hdc.system = HardDiskController::IDE;
868 hdc.idController = i.ulInstanceID;
869 hdc.strAddress = i.strAddress;
870 hdc.ulBusNumber = i.ulBusNumber;
871
872 vsys.mapControllers[i.ulInstanceID] = hdc;
873 }
874 break;
875
876 case OVFResourceType_ParallelSCSIHBA: // 6 SCSI controller
877 {
878 /* <Item>
879 <rasd:Caption>SCSI Controller 0 - LSI Logic</rasd:Caption>
880 <rasd:Description>SCI Controller</rasd:Description>
881 <rasd:ElementName>SCSI controller</rasd:ElementName>
882 <rasd:InstanceID>4</rasd:InstanceID>
883 <rasd:ResourceSubType>LsiLogic</rasd:ResourceSubType>
884 <rasd:ResourceType>6</rasd:ResourceType>
885 </Item> */
886 HardDiskController hdc;
887 hdc.system = HardDiskController::SCSI;
888 hdc.idController = i.ulInstanceID;
889 hdc.strControllerType = i.strResourceSubType;
890
891 vsys.mapControllers[i.ulInstanceID] = hdc;
892 }
893 break;
894
895 case OVFResourceType_EthernetAdapter: // 10
896 {
897 /* <Item>
898 <rasd:Caption>Ethernet adapter on 'Bridged'</rasd:Caption>
899 <rasd:AutomaticAllocation>true</rasd:AutomaticAllocation>
900 <rasd:Connection>Bridged</rasd:Connection>
901 <rasd:InstanceID>6</rasd:InstanceID>
902 <rasd:ResourceType>10</rasd:ResourceType>
903 <rasd:ResourceSubType>E1000</rasd:ResourceSubType>
904 </Item>
905
906 OVF spec DSP 0243 page 21:
907 "For an Ethernet adapter, this specifies the abstract network connection name
908 for the virtual machine. All Ethernet adapters that specify the same abstract
909 network connection name within an OVF package shall be deployed on the same
910 network. The abstract network connection name shall be listed in the NetworkSection
911 at the outermost envelope level." */
912
913 // only store the name
914 EthernetAdapter ea;
915 ea.strAdapterType = i.strResourceSubType;
916 ea.strNetworkName = i.strConnection;
917 vsys.llEthernetAdapters.push_back(ea);
918 }
919 break;
920
921 case OVFResourceType_FloppyDrive: // 14
922 vsys.fHasFloppyDrive = true; // we have no additional information
923 break;
924
925 case OVFResourceType_CDDrive: // 15
926 /* <Item ovf:required="false">
927 <rasd:Caption>cdrom1</rasd:Caption>
928 <rasd:InstanceId>7</rasd:InstanceId>
929 <rasd:ResourceType>15</rasd:ResourceType>
930 <rasd:AutomaticAllocation>true</rasd:AutomaticAllocation>
931 <rasd:Parent>5</rasd:Parent>
932 <rasd:AddressOnParent>0</rasd:AddressOnParent>
933 </Item> */
934 // I tried to see what happens if I set an ISO for the CD-ROM in VMware Workstation,
935 // but then the ovftool dies with "Device backing not supported". So I guess if
936 // VMware can't export ISOs, then we don't need to be able to import them right now.
937 vsys.fHasCdromDrive = true; // we have no additional information
938 break;
939
940 case OVFResourceType_HardDisk: // 17
941 {
942 /* <Item>
943 <rasd:Caption>Harddisk 1</rasd:Caption>
944 <rasd:Description>HD</rasd:Description>
945 <rasd:ElementName>Hard Disk</rasd:ElementName>
946 <rasd:HostResource>ovf://disk/lamp</rasd:HostResource>
947 <rasd:InstanceID>5</rasd:InstanceID>
948 <rasd:Parent>4</rasd:Parent>
949 <rasd:ResourceType>17</rasd:ResourceType>
950 </Item> */
951
952 // look up the hard disk controller element whose InstanceID equals our Parent;
953 // this is how the connection is specified in OVF
954 ControllersMap::const_iterator it = vsys.mapControllers.find(i.ulParent);
955 if (it == vsys.mapControllers.end())
956 return setError(VBOX_E_FILE_ERROR,
957 tr("Error reading \"%s\": Hard disk item with instance ID %d specifies invalid parent %d, line %d"),
958 pcszPath,
959 i.ulInstanceID,
960 i.ulParent,
961 i.ulLineNumber);
962 //const HardDiskController &hdc = it->second;
963
964 VirtualDisk vd;
965 vd.idController = i.ulParent;
966 i.strAddressOnParent.toInt(vd.ulAddressOnParent);
967 // ovf://disk/lamp
968 // 123456789012345
969 if (i.strHostResource.substr(0, 11) == "ovf://disk/")
970 vd.strDiskId = i.strHostResource.substr(11);
971 else if (i.strHostResource.substr(0, 6) == "/disk/")
972 vd.strDiskId = i.strHostResource.substr(6);
973
974 if ( !(vd.strDiskId.length())
975 || (m->mapDisks.find(vd.strDiskId) == m->mapDisks.end())
976 )
977 return setError(VBOX_E_FILE_ERROR,
978 tr("Error reading \"%s\": Hard disk item with instance ID %d specifies invalid host resource \"%s\", line %d"),
979 pcszPath,
980 i.ulInstanceID,
981 i.strHostResource.c_str(),
982 i.ulLineNumber);
983
984 vsys.mapVirtualDisks[vd.strDiskId] = vd;
985 }
986 break;
987
988 case OVFResourceType_OtherStorageDevice: // 20 SATA controller
989 {
990 /* <Item>
991 <rasd:Description>SATA Controller</rasd:Description>
992 <rasd:Caption>sataController0</rasd:Caption>
993 <rasd:InstanceID>4</rasd:InstanceID>
994 <rasd:ResourceType>20</rasd:ResourceType>
995 <rasd:ResourceSubType>AHCI</rasd:ResourceSubType>
996 <rasd:Address>0</rasd:Address>
997 <rasd:BusNumber>0</rasd:BusNumber>
998 </Item> */
999 if (i.strCaption.startsWith ("sataController", Utf8Str::CaseInsensitive) &&
1000 !i.strResourceSubType.compare ("AHCI", Utf8Str::CaseInsensitive))
1001 {
1002 HardDiskController hdc;
1003 hdc.system = HardDiskController::SATA;
1004 hdc.idController = i.ulInstanceID;
1005 hdc.strControllerType = i.strResourceSubType;
1006
1007 vsys.mapControllers[i.ulInstanceID] = hdc;
1008 }
1009 else
1010 return setError(VBOX_E_FILE_ERROR,
1011 tr("Error reading \"%s\": Host resource of type \"Other Storage Device (%d)\" is supported with SATA AHCI controllers only, line %d"),
1012 pcszPath,
1013 OVFResourceType_OtherStorageDevice,
1014 i.ulLineNumber);
1015 }
1016 break;
1017
1018 case OVFResourceType_USBController: // 23
1019 /* <Item ovf:required="false">
1020 <rasd:Caption>usb</rasd:Caption>
1021 <rasd:Description>USB Controller</rasd:Description>
1022 <rasd:InstanceId>3</rasd:InstanceId>
1023 <rasd:ResourceType>23</rasd:ResourceType>
1024 <rasd:Address>0</rasd:Address>
1025 <rasd:BusNumber>0</rasd:BusNumber>
1026 </Item> */
1027 vsys.fHasUsbController = true; // we have no additional information
1028 break;
1029
1030 case OVFResourceType_SoundCard: // 35
1031 /* <Item ovf:required="false">
1032 <rasd:Caption>sound</rasd:Caption>
1033 <rasd:Description>Sound Card</rasd:Description>
1034 <rasd:InstanceId>10</rasd:InstanceId>
1035 <rasd:ResourceType>35</rasd:ResourceType>
1036 <rasd:ResourceSubType>ensoniq1371</rasd:ResourceSubType>
1037 <rasd:AutomaticAllocation>false</rasd:AutomaticAllocation>
1038 <rasd:AddressOnParent>3</rasd:AddressOnParent>
1039 </Item> */
1040 vsys.strSoundCardType = i.strResourceSubType;
1041 break;
1042
1043 default:
1044 return setError(VBOX_E_FILE_ERROR,
1045 tr("Error reading \"%s\": Unknown resource type %d in hardware item, line %d"),
1046 pcszPath,
1047 i.resourceType,
1048 i.ulLineNumber);
1049 }
1050 }
1051 }
1052 else if ( (!strcmp(pcszElemName, "OperatingSystemSection"))
1053 || (!strcmp(pcszTypeAttr, "ovf:OperatingSystemSection_Type"))
1054 )
1055 {
1056 uint64_t cimos64;
1057 if (!(pelmThis->getAttributeValue("id", cimos64)))
1058 return setError(VBOX_E_FILE_ERROR,
1059 tr("Error reading \"%s\": missing or invalid 'ovf:id' attribute in operating system section element, line %d"),
1060 pcszPath,
1061 pelmThis->getLineNumber());
1062
1063 vsys.cimos = (CIMOSType_T)cimos64;
1064 }
1065 }
1066
1067 // now create the virtual system
1068 m->llVirtualSystems.push_back(vsys);
1069
1070 return S_OK;
1071}
1072
1073////////////////////////////////////////////////////////////////////////////////
1074//
1075// IAppliance public methods
1076//
1077////////////////////////////////////////////////////////////////////////////////
1078
1079/**
1080 * Public method implementation.
1081 * @param
1082 * @return
1083 */
1084STDMETHODIMP Appliance::COMGETTER(Path)(BSTR *aPath)
1085{
1086 if (!aPath)
1087 return E_POINTER;
1088
1089 AutoCaller autoCaller(this);
1090 CheckComRCReturnRC(autoCaller.rc());
1091
1092 AutoReadLock alock(this);
1093
1094 Bstr bstrPath(m->strPath);
1095 bstrPath.cloneTo(aPath);
1096
1097 return S_OK;
1098}
1099
1100/**
1101 * Public method implementation.
1102 * @param
1103 * @return
1104 */
1105STDMETHODIMP Appliance::COMGETTER(Disks)(ComSafeArrayOut(BSTR, aDisks))
1106{
1107 CheckComArgOutSafeArrayPointerValid(aDisks);
1108
1109 AutoCaller autoCaller(this);
1110 CheckComRCReturnRC(autoCaller.rc());
1111
1112 AutoReadLock alock(this);
1113
1114 size_t c = m->mapDisks.size();
1115 com::SafeArray<BSTR> sfaDisks(c);
1116
1117 DiskImagesMap::const_iterator it;
1118 size_t i = 0;
1119 for (it = m->mapDisks.begin();
1120 it != m->mapDisks.end();
1121 ++it, ++i)
1122 {
1123 // create a string representing this disk
1124 const DiskImage &d = it->second;
1125 char *psz = NULL;
1126 RTStrAPrintf(&psz,
1127 "%s\t"
1128 "%RI64\t"
1129 "%RI64\t"
1130 "%s\t"
1131 "%s\t"
1132 "%RI64\t"
1133 "%RI64\t"
1134 "%s",
1135 d.strDiskId.c_str(),
1136 d.iCapacity,
1137 d.iPopulatedSize,
1138 d.strFormat.c_str(),
1139 d.strHref.c_str(),
1140 d.iSize,
1141 d.iChunkSize,
1142 d.strCompression.c_str());
1143 Utf8Str utf(psz);
1144 Bstr bstr(utf);
1145 // push to safearray
1146 bstr.cloneTo(&sfaDisks[i]);
1147 RTStrFree(psz);
1148 }
1149
1150 sfaDisks.detachTo(ComSafeArrayOutArg(aDisks));
1151
1152 return S_OK;
1153}
1154
1155/**
1156 * Public method implementation.
1157 * @param
1158 * @return
1159 */
1160STDMETHODIMP Appliance::COMGETTER(VirtualSystemDescriptions)(ComSafeArrayOut(IVirtualSystemDescription*, aVirtualSystemDescriptions))
1161{
1162 CheckComArgOutSafeArrayPointerValid(aVirtualSystemDescriptions);
1163
1164 AutoCaller autoCaller(this);
1165 CheckComRCReturnRC(autoCaller.rc());
1166
1167 AutoReadLock alock(this);
1168
1169 SafeIfaceArray<IVirtualSystemDescription> sfaVSD(m->virtualSystemDescriptions);
1170 sfaVSD.detachTo(ComSafeArrayOutArg(aVirtualSystemDescriptions));
1171
1172 return S_OK;
1173}
1174
1175/**
1176 * Public method implementation.
1177 * @param path
1178 * @return
1179 */
1180STDMETHODIMP Appliance::Read(IN_BSTR path)
1181{
1182 HRESULT rc = S_OK;
1183
1184 if (!path)
1185 return E_POINTER;
1186
1187 AutoCaller autoCaller(this);
1188 CheckComRCReturnRC(autoCaller.rc());
1189
1190 AutoWriteLock alock(this);
1191
1192 // see if we can handle this file; for now we insist it has an ".ovf" extension
1193 m->strPath = path;
1194 if (!m->strPath.endsWith(".ovf", Utf8Str::CaseInsensitive))
1195 return setError(VBOX_E_FILE_ERROR,
1196 tr("Appliance file must have .ovf extension"));
1197
1198 try
1199 {
1200 xml::XmlFileParser parser;
1201 xml::Document doc;
1202 parser.read(m->strPath.raw(),
1203 doc);
1204
1205 const xml::ElementNode *pRootElem = doc.getRootElement();
1206 if (strcmp(pRootElem->getName(), "Envelope"))
1207 return setError(VBOX_E_FILE_ERROR,
1208 tr("Root element in OVF file must be \"Envelope\"."));
1209
1210 // OVF has the following rough layout:
1211 /*
1212 -- <References> .... files referenced from other parts of the file, such as VMDK images
1213 -- Metadata, comprised of several section commands
1214 -- virtual machines, either a single <VirtualSystem>, or a <VirtualSystemCollection>
1215 -- optionally <Strings> for localization
1216 */
1217
1218 // get all "File" child elements of "References" section so we can look up files easily;
1219 // first find the "References" sections so we can look up files
1220 xml::ElementNodesList listFileElements; // receives all /Envelope/References/File nodes
1221 const xml::ElementNode *pReferencesElem;
1222 if ((pReferencesElem = pRootElem->findChildElement("References")))
1223 pReferencesElem->getChildElements(listFileElements, "File");
1224
1225 // now go though the sections
1226 if (!(SUCCEEDED(rc = LoopThruSections(m->strPath.raw(), pReferencesElem, pRootElem))))
1227 return rc;
1228 }
1229 catch(xml::Error &x)
1230 {
1231 return setError(VBOX_E_FILE_ERROR,
1232 x.what());
1233 }
1234
1235 return S_OK;
1236}
1237
1238/**
1239 * Public method implementation.
1240 * @return
1241 */
1242STDMETHODIMP Appliance::Interpret()
1243{
1244 // @todo:
1245 // - don't use COM methods but the methods directly (faster, but needs appropriate locking of that objects itself (s. HardDisk))
1246 // - Appropriate handle errors like not supported file formats
1247 AutoCaller autoCaller(this);
1248 CheckComRCReturnRC(autoCaller.rc());
1249
1250 AutoWriteLock(this);
1251
1252 HRESULT rc = S_OK;
1253
1254 /* Clear any previous virtual system descriptions */
1255 m->virtualSystemDescriptions.clear();
1256
1257 /* We need the default path for storing disk images */
1258 ComPtr<ISystemProperties> systemProps;
1259 rc = mVirtualBox->COMGETTER(SystemProperties)(systemProps.asOutParam());
1260 CheckComRCReturnRC(rc);
1261 Bstr bstrDefaultHardDiskLocation;
1262 rc = systemProps->COMGETTER(DefaultHardDiskFolder)(bstrDefaultHardDiskLocation.asOutParam());
1263 CheckComRCReturnRC(rc);
1264
1265 /* Try/catch so we can clean up on error */
1266 try
1267 {
1268 list<VirtualSystem>::const_iterator it;
1269 /* Iterate through all virtual systems */
1270 for (it = m->llVirtualSystems.begin();
1271 it != m->llVirtualSystems.end();
1272 ++it)
1273 {
1274 const VirtualSystem &vsysThis = *it;
1275
1276 ComObjPtr<VirtualSystemDescription> pNewDesc;
1277 rc = pNewDesc.createObject();
1278 CheckComRCThrowRC(rc);
1279 rc = pNewDesc->init();
1280 CheckComRCThrowRC(rc);
1281
1282 /* Guest OS type */
1283 Utf8Str strOsTypeVBox,
1284 strCIMOSType = Utf8StrFmt("%RI32", (uint32_t)vsysThis.cimos);
1285 convertCIMOSType2VBoxOSType(strOsTypeVBox, vsysThis.cimos);
1286 pNewDesc->addEntry(VirtualSystemDescriptionType_OS,
1287 "",
1288 strCIMOSType,
1289 strOsTypeVBox);
1290
1291 /* VM name */
1292 /* If the there isn't any name specified create a default one out of
1293 * the OS type */
1294 Utf8Str nameVBox = vsysThis.strName;
1295 if (nameVBox.isEmpty())
1296 nameVBox = strOsTypeVBox;
1297 searchUniqueVMName(nameVBox);
1298 pNewDesc->addEntry(VirtualSystemDescriptionType_Name,
1299 "",
1300 vsysThis.strName,
1301 nameVBox);
1302
1303 if (!vsysThis.strDescription.isEmpty())
1304 pNewDesc->addEntry(VirtualSystemDescriptionType_Description,
1305 "",
1306 vsysThis.strDescription,
1307 vsysThis.strDescription);
1308
1309 /* Now that we know the OS type, get our internal defaults based on that. */
1310 ComPtr<IGuestOSType> pGuestOSType;
1311 rc = mVirtualBox->GetGuestOSType(Bstr(strOsTypeVBox), pGuestOSType.asOutParam());
1312 CheckComRCThrowRC(rc);
1313
1314 /* CPU count */
1315 ULONG cpuCountVBox = vsysThis.cCPUs;
1316 /* Check for the constrains */
1317 if (cpuCountVBox > 1) //SchemaDefs::MaxCPUCount)
1318 {
1319 addWarning(tr("The virtual system \"%s\" claims support for %u CPU's, but VirtualBox has support for max %u CPU's only."),
1320 vsysThis.strName.c_str(), cpuCountVBox, 1); //SchemaDefs::MaxCPUCount);
1321 cpuCountVBox = 1; //SchemaDefs::MaxCPUCount;
1322 }
1323 if (vsysThis.cCPUs == 0)
1324 cpuCountVBox = 1;
1325 pNewDesc->addEntry(VirtualSystemDescriptionType_CPU,
1326 "",
1327 Utf8StrFmt("%RI32", (uint32_t)vsysThis.cCPUs),
1328 Utf8StrFmt("%RI32", (uint32_t)cpuCountVBox));
1329
1330 /* RAM */
1331 uint64_t ullMemSizeVBox = vsysThis.ullMemorySize / _1M;
1332 /* Check for the constrains */
1333 if (ullMemSizeVBox != 0 &&
1334 (ullMemSizeVBox < static_cast<uint64_t>(SchemaDefs::MinGuestRAM) ||
1335 ullMemSizeVBox > static_cast<uint64_t>(SchemaDefs::MaxGuestRAM)))
1336 {
1337 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."),
1338 vsysThis.strName.c_str(), ullMemSizeVBox, SchemaDefs::MinGuestRAM, SchemaDefs::MaxGuestRAM);
1339 ullMemSizeVBox = RT_MIN(RT_MAX(ullMemSizeVBox, static_cast<uint64_t>(SchemaDefs::MinGuestRAM)), static_cast<uint64_t>(SchemaDefs::MaxGuestRAM));
1340 }
1341 if (vsysThis.ullMemorySize == 0)
1342 {
1343 /* If the RAM of the OVF is zero, use our predefined values */
1344 ULONG memSizeVBox2;
1345 rc = pGuestOSType->COMGETTER(RecommendedRAM)(&memSizeVBox2);
1346 CheckComRCThrowRC(rc);
1347 /* VBox stores that in MByte */
1348 ullMemSizeVBox = (uint64_t)memSizeVBox2;
1349 }
1350 pNewDesc->addEntry(VirtualSystemDescriptionType_Memory,
1351 "",
1352 Utf8StrFmt("%RI64", (uint64_t)vsysThis.ullMemorySize),
1353 Utf8StrFmt("%RI64", (uint64_t)ullMemSizeVBox));
1354
1355 /* Audio */
1356 if (!vsysThis.strSoundCardType.isNull())
1357 /* Currently we set the AC97 always.
1358 @todo: figure out the hardware which could be possible */
1359 pNewDesc->addEntry(VirtualSystemDescriptionType_SoundCard,
1360 "",
1361 vsysThis.strSoundCardType,
1362 Utf8StrFmt("%RI32", (uint32_t)AudioControllerType_AC97));
1363
1364#ifdef VBOX_WITH_USB
1365 /* USB Controller */
1366 if (vsysThis.fHasUsbController)
1367 pNewDesc->addEntry(VirtualSystemDescriptionType_USBController, "", "", "");
1368#endif /* VBOX_WITH_USB */
1369
1370 /* Network Controller */
1371 uint32_t cEthernetAdapters = vsysThis.llEthernetAdapters.size();
1372 if (cEthernetAdapters > 0)
1373 {
1374 /* Check for the constrains */
1375 if (cEthernetAdapters > SchemaDefs::NetworkAdapterCount)
1376 addWarning(tr("The virtual system \"%s\" claims support for %u network adapters, but VirtualBox has support for max %u network adapter only."),
1377 vsysThis.strName.c_str(), cEthernetAdapters, SchemaDefs::NetworkAdapterCount);
1378
1379 /* Get the default network adapter type for the selected guest OS */
1380 NetworkAdapterType_T defaultAdapterVBox = NetworkAdapterType_Am79C970A;
1381 rc = pGuestOSType->COMGETTER(AdapterType)(&defaultAdapterVBox);
1382 CheckComRCThrowRC(rc);
1383
1384 EthernetAdaptersList::const_iterator itEA;
1385 /* Iterate through all abstract networks. We support 8 network
1386 * adapters at the maximum, so the first 8 will be added only. */
1387 size_t a = 0;
1388 for (itEA = vsysThis.llEthernetAdapters.begin();
1389 itEA != vsysThis.llEthernetAdapters.end() && a < SchemaDefs::NetworkAdapterCount;
1390 ++itEA, ++a)
1391 {
1392 const EthernetAdapter &ea = *itEA; // logical network to connect to
1393 Utf8Str strNetwork = ea.strNetworkName;
1394 // make sure it's one of these two
1395 if ( (strNetwork.compare("Null", Utf8Str::CaseInsensitive))
1396 && (strNetwork.compare("Bridged", Utf8Str::CaseInsensitive))
1397 && (strNetwork.compare("Internal", Utf8Str::CaseInsensitive))
1398 && (strNetwork.compare("HostOnly", Utf8Str::CaseInsensitive))
1399 )
1400 strNetwork = "NAT";
1401
1402 /* Figure out the hardware type */
1403 NetworkAdapterType_T nwAdapterVBox = defaultAdapterVBox;
1404 if (!ea.strAdapterType.compare("PCNet32", Utf8Str::CaseInsensitive))
1405 {
1406 /* If the default adapter is already one of the two
1407 * PCNet adapters use the default one. If not use the
1408 * Am79C970A as fallback. */
1409 if (!(defaultAdapterVBox == NetworkAdapterType_Am79C970A ||
1410 defaultAdapterVBox == NetworkAdapterType_Am79C973))
1411 nwAdapterVBox = NetworkAdapterType_Am79C970A;
1412 }
1413#ifdef VBOX_WITH_E1000
1414 else if (!ea.strAdapterType.compare("E1000", Utf8Str::CaseInsensitive))
1415 {
1416 /* If the default adapter is already one of the two
1417 * E1000 adapters use the default one. If not use the
1418 * I82540EM as fallback. */
1419 if (!(defaultAdapterVBox == NetworkAdapterType_I82540EM ||
1420 defaultAdapterVBox == NetworkAdapterType_I82543GC))
1421 nwAdapterVBox = NetworkAdapterType_I82540EM;
1422 }
1423#endif /* VBOX_WITH_E1000 */
1424
1425 pNewDesc->addEntry(VirtualSystemDescriptionType_NetworkAdapter,
1426 "", // ref
1427 strNetwork, // orig
1428 Utf8StrFmt("%RI32", (uint32_t)nwAdapterVBox), // conf
1429 Utf8StrFmt("type=%s", strNetwork.c_str())); // extra conf
1430 }
1431 }
1432
1433 /* Floppy Drive */
1434 if (vsysThis.fHasFloppyDrive)
1435 pNewDesc->addEntry(VirtualSystemDescriptionType_Floppy, "", "", "");
1436
1437 /* CD Drive */
1438 /* @todo: I can't disable the CDROM. So nothing to do for now */
1439 /*
1440 if (vsysThis.fHasCdromDrive)
1441 pNewDesc->addEntry(VirtualSystemDescriptionType_CDROM, "", "", "");*/
1442
1443 /* Hard disk Controller */
1444 uint16_t cIDEused = 0;
1445 uint16_t cSATAused = 0;
1446 uint16_t cSCSIused = 0;
1447 ControllersMap::const_iterator hdcIt;
1448 /* Iterate through all hard disk controllers */
1449 for (hdcIt = vsysThis.mapControllers.begin();
1450 hdcIt != vsysThis.mapControllers.end();
1451 ++hdcIt)
1452 {
1453 const HardDiskController &hdc = hdcIt->second;
1454 Utf8Str strControllerID = Utf8StrFmt("%RI32", (uint32_t)hdc.idController);
1455
1456 switch (hdc.system)
1457 {
1458 case HardDiskController::IDE:
1459 {
1460 /* Check for the constrains */
1461 /* @todo: I'm very confused! Are these bits *one* controller or
1462 is every port/bus declared as an extra controller. */
1463 if (cIDEused < 4)
1464 {
1465 // @todo: figure out the IDE types
1466 /* Use PIIX4 as default */
1467 Utf8Str strType = "PIIX4";
1468 if (!hdc.strControllerType.compare("PIIX3", Utf8Str::CaseInsensitive))
1469 strType = "PIIX3";
1470 else if (!hdc.strControllerType.compare("ICH6", Utf8Str::CaseInsensitive))
1471 strType = "ICH6";
1472 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerIDE,
1473 strControllerID,
1474 hdc.strControllerType,
1475 strType);
1476 }
1477 else
1478 {
1479 /* Warn only once */
1480 if (cIDEused == 1)
1481 addWarning(tr("The virtual \"%s\" system requests support for more than one IDE controller, but VirtualBox has support for only one."),
1482 vsysThis.strName.c_str());
1483
1484 }
1485 ++cIDEused;
1486 break;
1487 }
1488
1489#ifdef VBOX_WITH_AHCI
1490 case HardDiskController::SATA:
1491 {
1492 /* Check for the constrains */
1493 if (cSATAused < 1)
1494 {
1495 // @todo: figure out the SATA types
1496 /* We only support a plain AHCI controller, so use them always */
1497 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerSATA,
1498 strControllerID,
1499 hdc.strControllerType,
1500 "AHCI");
1501 }
1502 else
1503 {
1504 /* Warn only once */
1505 if (cSATAused == 1)
1506 addWarning(tr("The virtual system \"%s\" requests support for more than one SATA controller, but VirtualBox has support for only one"),
1507 vsysThis.strName.c_str());
1508
1509 }
1510 ++cSATAused;
1511 break;
1512 }
1513#endif /* VBOX_WITH_AHCI */
1514
1515 case HardDiskController::SCSI:
1516 {
1517 /* Check for the constrains */
1518 if (cSCSIused < 1)
1519 {
1520 Utf8Str hdcController = "LsiLogic";
1521 if (!hdc.strControllerType.compare("BusLogic", Utf8Str::CaseInsensitive))
1522 hdcController = "BusLogic";
1523 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerSCSI,
1524 strControllerID,
1525 hdc.strControllerType,
1526 hdcController);
1527 }
1528 else
1529 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."),
1530 vsysThis.strName.c_str(),
1531 hdc.strControllerType.c_str(),
1532 strControllerID.c_str());
1533 ++cSCSIused;
1534 break;
1535 }
1536 }
1537 }
1538
1539 /* Hard disks */
1540 if (vsysThis.mapVirtualDisks.size() > 0)
1541 {
1542 VirtualDisksMap::const_iterator itVD;
1543 /* Iterate through all hard disks ()*/
1544 for (itVD = vsysThis.mapVirtualDisks.begin();
1545 itVD != vsysThis.mapVirtualDisks.end();
1546 ++itVD)
1547 {
1548 const VirtualDisk &hd = itVD->second;
1549 /* Get the associated disk image */
1550 const DiskImage &di = m->mapDisks[hd.strDiskId];
1551
1552 // @todo:
1553 // - figure out all possible vmdk formats we also support
1554 // - figure out if there is a url specifier for vhd already
1555 // - we need a url specifier for the vdi format
1556 if ( di.strFormat.compare("http://www.vmware.com/specifications/vmdk.html#sparse", Utf8Str::CaseInsensitive)
1557 || di.strFormat.compare("http://www.vmware.com/specifications/vmdk.html#compressed", Utf8Str::CaseInsensitive))
1558 {
1559 /* If the href is empty use the VM name as filename */
1560 Utf8Str strFilename = di.strHref;
1561 if (!strFilename.length())
1562 strFilename = Utf8StrFmt("%s.vmdk", nameVBox.c_str());
1563 /* Construct a unique target path */
1564 Utf8StrFmt strPath("%ls%c%s",
1565 bstrDefaultHardDiskLocation.raw(),
1566 RTPATH_DELIMITER,
1567 strFilename.c_str());
1568 searchUniqueDiskImageFilePath(strPath);
1569
1570 /* find the description for the hard disk controller
1571 * that has the same ID as hd.idController */
1572 const VirtualSystemDescriptionEntry *pController;
1573 if (!(pController = pNewDesc->findControllerFromID(hd.idController)))
1574 throw setError(E_FAIL,
1575 tr("Cannot find hard disk controller with OVF instance ID %RI32 to which disk \"%s\" should be attached"),
1576 hd.idController,
1577 di.strHref.c_str());
1578
1579 /* controller to attach to, and the bus within that controller */
1580 Utf8StrFmt strExtraConfig("controller=%RI16;channel=%RI16",
1581 pController->ulIndex,
1582 hd.ulAddressOnParent);
1583 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskImage,
1584 hd.strDiskId,
1585 di.strHref,
1586 strPath,
1587 strExtraConfig);
1588 }
1589 else
1590 throw setError(VBOX_E_FILE_ERROR,
1591 tr("Unsupported format for virtual disk image in OVF: \"%s\"", di.strFormat.c_str()));
1592 }
1593 }
1594
1595 m->virtualSystemDescriptions.push_back(pNewDesc);
1596 }
1597 }
1598 catch (HRESULT aRC)
1599 {
1600 /* On error we clear the list & return */
1601 m->virtualSystemDescriptions.clear();
1602 rc = aRC;
1603 }
1604
1605 return rc;
1606}
1607
1608/**
1609 * Public method implementation.
1610 * @param aProgress
1611 * @return
1612 */
1613STDMETHODIMP Appliance::ImportMachines(IProgress **aProgress)
1614{
1615 CheckComArgOutPointerValid(aProgress);
1616
1617 AutoCaller autoCaller(this);
1618 CheckComRCReturnRC(autoCaller.rc());
1619
1620 AutoReadLock(this);
1621
1622 HRESULT rc = S_OK;
1623
1624 ComObjPtr<Progress> progress;
1625 try
1626 {
1627 uint32_t opCount = calcMaxProgress();
1628 Bstr progressDesc = BstrFmt(tr("Import appliance '%s'"),
1629 m->strPath.raw());
1630 /* Create the progress object */
1631 progress.createObject();
1632 rc = progress->init(mVirtualBox, static_cast<IAppliance*>(this),
1633 progressDesc,
1634 FALSE /* aCancelable */,
1635 opCount,
1636 progressDesc);
1637 if (FAILED(rc)) throw rc;
1638
1639 /* Initialize our worker task */
1640 std::auto_ptr<TaskImportMachines> task(new TaskImportMachines(this, progress));
1641 //AssertComRCThrowRC (task->autoCaller.rc());
1642
1643 rc = task->startThread();
1644 if (FAILED(rc)) throw rc;
1645
1646 task.release();
1647 }
1648 catch (HRESULT aRC)
1649 {
1650 rc = aRC;
1651 }
1652
1653 if (SUCCEEDED(rc))
1654 /* Return progress to the caller */
1655 progress.queryInterfaceTo(aProgress);
1656
1657 return rc;
1658}
1659
1660STDMETHODIMP Appliance::Write(IN_BSTR path, IProgress **aProgress)
1661{
1662 HRESULT rc = S_OK;
1663
1664 CheckComArgOutPointerValid(aProgress);
1665
1666 AutoCaller autoCaller(this);
1667 if (FAILED(rc = autoCaller.rc())) return rc;
1668
1669 AutoWriteLock(this);
1670
1671 // see if we can handle this file; for now we insist it has an ".ovf" extension
1672 m->strPath = path;
1673 if (!m->strPath.endsWith(".ovf", Utf8Str::CaseInsensitive))
1674 return setError(VBOX_E_FILE_ERROR,
1675 tr("Appliance file must have .ovf extension"));
1676
1677 ComObjPtr<Progress> progress;
1678 try
1679 {
1680 uint32_t opCount = calcMaxProgress();
1681 Bstr progressDesc = BstrFmt(tr("Write appliance '%s'"),
1682 m->strPath.raw());
1683 /* Create the progress object */
1684 progress.createObject();
1685 rc = progress->init(mVirtualBox, static_cast<IAppliance*>(this),
1686 progressDesc,
1687 FALSE /* aCancelable */,
1688 opCount,
1689 progressDesc);
1690 CheckComRCThrowRC(rc);
1691
1692 /* Initialize our worker task */
1693 std::auto_ptr<TaskWriteOVF> task(new TaskWriteOVF(this, progress));
1694 //AssertComRCThrowRC (task->autoCaller.rc());
1695
1696 rc = task->startThread();
1697 CheckComRCThrowRC(rc);
1698
1699 task.release();
1700 }
1701 catch (HRESULT aRC)
1702 {
1703 rc = aRC;
1704 }
1705
1706 if (SUCCEEDED(rc))
1707 /* Return progress to the caller */
1708 progress.queryInterfaceTo(aProgress);
1709
1710 return rc;
1711}
1712
1713/**
1714* Public method implementation.
1715 * @return
1716 */
1717STDMETHODIMP Appliance::GetWarnings(ComSafeArrayOut(BSTR, aWarnings))
1718{
1719 if (ComSafeArrayOutIsNull(aWarnings))
1720 return E_POINTER;
1721
1722 AutoCaller autoCaller(this);
1723 CheckComRCReturnRC(autoCaller.rc());
1724
1725 AutoReadLock alock(this);
1726
1727 com::SafeArray<BSTR> sfaWarnings(m->llWarnings.size());
1728
1729 list<Utf8Str>::const_iterator it;
1730 size_t i = 0;
1731 for (it = m->llWarnings.begin();
1732 it != m->llWarnings.end();
1733 ++it, ++i)
1734 {
1735 Bstr bstr = *it;
1736 bstr.cloneTo(&sfaWarnings[i]);
1737 }
1738
1739 sfaWarnings.detachTo(ComSafeArrayOutArg(aWarnings));
1740
1741 return S_OK;
1742}
1743
1744HRESULT Appliance::searchUniqueVMName(Utf8Str& aName) const
1745{
1746 IMachine *machine = NULL;
1747 char *tmpName = RTStrDup(aName.c_str());
1748 int i = 1;
1749 /* @todo: Maybe too cost-intensive; try to find a lighter way */
1750 while (mVirtualBox->FindMachine(Bstr(tmpName), &machine) != VBOX_E_OBJECT_NOT_FOUND)
1751 {
1752 RTStrFree(tmpName);
1753 RTStrAPrintf(&tmpName, "%s_%d", aName.c_str(), i);
1754 ++i;
1755 }
1756 aName = tmpName;
1757 RTStrFree(tmpName);
1758
1759 return S_OK;
1760}
1761
1762HRESULT Appliance::searchUniqueDiskImageFilePath(Utf8Str& aName) const
1763{
1764 IHardDisk *harddisk = NULL;
1765 char *tmpName = RTStrDup(aName.c_str());
1766 int i = 1;
1767 /* Check if the file exists or if a file with this path is registered
1768 * already */
1769 /* @todo: Maybe too cost-intensive; try to find a lighter way */
1770 while (RTPathExists(tmpName) ||
1771 mVirtualBox->FindHardDisk(Bstr(tmpName), &harddisk) != VBOX_E_OBJECT_NOT_FOUND)
1772 {
1773 RTStrFree(tmpName);
1774 char *tmpDir = RTStrDup(aName.c_str());
1775 RTPathStripFilename(tmpDir);;
1776 char *tmpFile = RTStrDup(RTPathFilename(aName.c_str()));
1777 RTPathStripExt(tmpFile);
1778 const char *tmpExt = RTPathExt(aName.c_str());
1779 RTStrAPrintf(&tmpName, "%s%c%s_%d%s", tmpDir, RTPATH_DELIMITER, tmpFile, i, tmpExt);
1780 RTStrFree(tmpFile);
1781 RTStrFree(tmpDir);
1782 ++i;
1783 }
1784 aName = tmpName;
1785 RTStrFree(tmpName);
1786
1787 return S_OK;
1788}
1789
1790/**
1791 * Calculates the maximum progress value for importMachines() and write().
1792 * @return
1793 */
1794uint32_t Appliance::calcMaxProgress()
1795{
1796 /* Figure out how many sub operation the import will need */
1797 /* One for the appliance */
1798 uint32_t opCount = 1;
1799 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it;
1800 for (it = m->virtualSystemDescriptions.begin();
1801 it != m->virtualSystemDescriptions.end();
1802 ++it)
1803 {
1804 /* One for every Virtual System */
1805 ++opCount;
1806 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it);
1807 /* One for every hard disk of the Virtual System */
1808 std::list<VirtualSystemDescriptionEntry*> avsdeHDs = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskImage);
1809 opCount += (uint32_t)avsdeHDs.size();
1810 }
1811
1812 return opCount;
1813}
1814
1815void Appliance::addWarning(const char* aWarning, ...)
1816{
1817 va_list args;
1818 va_start(args, aWarning);
1819 Utf8StrFmtVA str(aWarning, args);
1820 va_end(args);
1821 m->llWarnings.push_back(str);
1822}
1823
1824struct MyHardDiskAttachment
1825{
1826 Guid uuid;
1827 ComPtr<IMachine> pMachine;
1828 Bstr controllerType;
1829 int32_t lChannel;
1830 int32_t lDevice;
1831};
1832
1833/**
1834 * Worker thread implementation for ImportMachines().
1835 * @param aThread
1836 * @param pvUser
1837 */
1838/* static */
1839DECLCALLBACK(int) Appliance::taskThreadImportMachines(RTTHREAD /* aThread */, void *pvUser)
1840{
1841 std::auto_ptr<TaskImportMachines> task(static_cast<TaskImportMachines*>(pvUser));
1842 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
1843
1844 Appliance *pAppliance = task->pAppliance;
1845
1846 LogFlowFuncEnter();
1847 LogFlowFunc(("Appliance %p\n", pAppliance));
1848
1849 AutoCaller autoCaller(pAppliance);
1850 CheckComRCReturnRC(autoCaller.rc());
1851
1852 AutoWriteLock appLock(pAppliance);
1853
1854 HRESULT rc = S_OK;
1855
1856 ComPtr<IVirtualBox> pVirtualBox(pAppliance->mVirtualBox);
1857
1858 // rollback for errors:
1859 // 1) a list of images that we created/imported
1860 list<MyHardDiskAttachment> llHardDiskAttachments;
1861 list< ComPtr<IHardDisk> > llHardDisksCreated;
1862 list<Guid> llMachinesRegistered;
1863
1864 ComPtr<ISession> session;
1865 bool fSessionOpen = false;
1866 rc = session.createInprocObject(CLSID_Session);
1867 CheckComRCReturnRC(rc);
1868
1869 list<VirtualSystem>::const_iterator it;
1870 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it1;
1871 /* Iterate through all virtual systems of that appliance */
1872 size_t i = 0;
1873 for (it = pAppliance->m->llVirtualSystems.begin(),
1874 it1 = pAppliance->m->virtualSystemDescriptions.begin();
1875 it != pAppliance->m->llVirtualSystems.end();
1876 ++it, ++it1, ++i)
1877 {
1878 const VirtualSystem &vsysThis = *it;
1879 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it1);
1880
1881 ComPtr<IMachine> pNewMachine;
1882
1883 /* Catch possible errors */
1884 try
1885 {
1886 if (!task->progress.isNull())
1887 task->progress->advanceOperation(BstrFmt(tr("Importing Virtual System %d"), i + 1));
1888
1889 /* How many sub notifications are necessary? */
1890 const float opCountMax = 100.0/5;
1891 uint32_t opCount = 0;
1892
1893 /* Guest OS type */
1894 std::list<VirtualSystemDescriptionEntry*> vsdeOS;
1895 vsdeOS = vsdescThis->findByType(VirtualSystemDescriptionType_OS);
1896 if (vsdeOS.size() < 1)
1897 throw setError(VBOX_E_FILE_ERROR,
1898 tr("Missing guest OS type"));
1899 const Utf8Str &strOsTypeVBox = vsdeOS.front()->strVbox;
1900
1901 /* Now that we know the base system get our internal defaults based on that. */
1902 ComPtr<IGuestOSType> osType;
1903 rc = pVirtualBox->GetGuestOSType(Bstr(strOsTypeVBox), osType.asOutParam());
1904 if (FAILED(rc)) throw rc;
1905
1906 /* Create the machine */
1907 /* First get the name */
1908 std::list<VirtualSystemDescriptionEntry*> vsdeName = vsdescThis->findByType(VirtualSystemDescriptionType_Name);
1909 if (vsdeName.size() < 1)
1910 throw setError(VBOX_E_FILE_ERROR,
1911 tr("Missing VM name"));
1912 const Utf8Str &strNameVBox = vsdeName.front()->strVbox;
1913 rc = pVirtualBox->CreateMachine(Bstr(strNameVBox), Bstr(strOsTypeVBox),
1914 Bstr(), Guid(),
1915 pNewMachine.asOutParam());
1916 if (FAILED(rc)) throw rc;
1917
1918 // and the description
1919 std::list<VirtualSystemDescriptionEntry*> vsdeDescription = vsdescThis->findByType(VirtualSystemDescriptionType_Description);
1920 if (vsdeDescription.size())
1921 {
1922 const Utf8Str &strDescription = vsdeDescription.front()->strVbox;
1923 rc = pNewMachine->COMSETTER(Description)(Bstr(strDescription));
1924 if (FAILED(rc)) throw rc;
1925 }
1926
1927 if (!task->progress.isNull())
1928 rc = task->progress->notifyProgress((uint32_t)(opCountMax * opCount++));
1929
1930 /* CPU count (ignored for now) */
1931 // EntriesList vsdeCPU = vsd->findByType (VirtualSystemDescriptionType_CPU);
1932
1933 /* RAM */
1934 std::list<VirtualSystemDescriptionEntry*> vsdeRAM = vsdescThis->findByType(VirtualSystemDescriptionType_Memory);
1935 ComAssertMsgThrow(vsdeRAM.size() == 1, ("RAM size missing"), E_FAIL);
1936 const Utf8Str &memoryVBox = vsdeRAM.front()->strVbox;
1937 ULONG tt = (ULONG)RTStrToUInt64(memoryVBox.c_str());
1938 rc = pNewMachine->COMSETTER(MemorySize)(tt);
1939 if (FAILED(rc)) throw rc;
1940
1941 /* VRAM */
1942 /* Get the recommended VRAM for this guest OS type */
1943 ULONG vramVBox;
1944 rc = osType->COMGETTER(RecommendedVRAM)(&vramVBox);
1945 if (FAILED(rc)) throw rc;
1946
1947 /* Set the VRAM */
1948 rc = pNewMachine->COMSETTER(VRAMSize)(vramVBox);
1949 if (FAILED(rc)) throw rc;
1950
1951 if (!task->progress.isNull())
1952 task->progress->notifyProgress((uint32_t)(opCountMax * opCount++));
1953
1954 /* Audio Adapter */
1955 std::list<VirtualSystemDescriptionEntry*> vsdeAudioAdapter = vsdescThis->findByType(VirtualSystemDescriptionType_SoundCard);
1956 /* @todo: we support one audio adapter only */
1957 if (vsdeAudioAdapter.size() > 0)
1958 {
1959 const Utf8Str& audioAdapterVBox = vsdeAudioAdapter.front()->strVbox;
1960 if (audioAdapterVBox.compare("null", Utf8Str::CaseInsensitive) != 0)
1961 {
1962 uint32_t audio = RTStrToUInt32(audioAdapterVBox.c_str());
1963 ComPtr<IAudioAdapter> audioAdapter;
1964 rc = pNewMachine->COMGETTER(AudioAdapter)(audioAdapter.asOutParam());
1965 if (FAILED(rc)) throw rc;
1966 rc = audioAdapter->COMSETTER(Enabled)(true);
1967 if (FAILED(rc)) throw rc;
1968 rc = audioAdapter->COMSETTER(AudioController)(static_cast<AudioControllerType_T>(audio));
1969 if (FAILED(rc)) throw rc;
1970 }
1971 }
1972
1973#ifdef VBOX_WITH_USB
1974 /* USB Controller */
1975 std::list<VirtualSystemDescriptionEntry*> vsdeUSBController = vsdescThis->findByType(VirtualSystemDescriptionType_USBController);
1976 // USB support is enabled if there's at least one such entry; to disable USB support,
1977 // the type of the USB item would have been changed to "ignore"
1978 bool fUSBEnabled = vsdeUSBController.size() > 0;
1979
1980 ComPtr<IUSBController> usbController;
1981 rc = pNewMachine->COMGETTER(USBController)(usbController.asOutParam());
1982 if (FAILED(rc)) throw rc;
1983 rc = usbController->COMSETTER(Enabled)(fUSBEnabled);
1984 if (FAILED(rc)) throw rc;
1985#endif /* VBOX_WITH_USB */
1986
1987 if (!task->progress.isNull())
1988 task->progress->notifyProgress((uint32_t)(opCountMax * opCount++));
1989
1990 /* Change the network adapters */
1991 std::list<VirtualSystemDescriptionEntry*> vsdeNW = vsdescThis->findByType(VirtualSystemDescriptionType_NetworkAdapter);
1992 if (vsdeNW.size() == 0)
1993 {
1994 /* No network adapters, so we have to disable our default one */
1995 ComPtr<INetworkAdapter> nwVBox;
1996 rc = pNewMachine->GetNetworkAdapter(0, nwVBox.asOutParam());
1997 if (FAILED(rc)) throw rc;
1998 rc = nwVBox->COMSETTER(Enabled)(false);
1999 if (FAILED(rc)) throw rc;
2000 }
2001 else
2002 {
2003 list<VirtualSystemDescriptionEntry*>::const_iterator nwIt;
2004 /* Iterate through all network cards. We support 8 network adapters
2005 * at the maximum. (@todo: warn if there are more!) */
2006 size_t a = 0;
2007 for (nwIt = vsdeNW.begin();
2008 (nwIt != vsdeNW.end() && a < SchemaDefs::NetworkAdapterCount);
2009 ++nwIt, ++a)
2010 {
2011 const VirtualSystemDescriptionEntry* pvsys = *nwIt;
2012
2013 const Utf8Str &nwTypeVBox = pvsys->strVbox;
2014 uint32_t tt1 = RTStrToUInt32(nwTypeVBox.c_str());
2015 ComPtr<INetworkAdapter> pNetworkAdapter;
2016 rc = pNewMachine->GetNetworkAdapter((ULONG)a, pNetworkAdapter.asOutParam());
2017 if (FAILED(rc)) throw rc;
2018 /* Enable the network card & set the adapter type */
2019 rc = pNetworkAdapter->COMSETTER(Enabled)(true);
2020 if (FAILED(rc)) throw rc;
2021 rc = pNetworkAdapter->COMSETTER(AdapterType)(static_cast<NetworkAdapterType_T>(tt1));
2022 if (FAILED(rc)) throw rc;
2023
2024 // default is NAT; change to "bridged" if extra conf says so
2025 if (!pvsys->strExtraConfig.compare("type=Bridged", Utf8Str::CaseInsensitive))
2026 {
2027 /* Attach to the right interface */
2028 rc = pNetworkAdapter->AttachToBridgedInterface();
2029 if (FAILED(rc)) throw rc;
2030 ComPtr<IHost> host;
2031 rc = pVirtualBox->COMGETTER(Host)(host.asOutParam());
2032 if (FAILED(rc)) throw rc;
2033 com::SafeIfaceArray<IHostNetworkInterface> nwInterfaces;
2034 rc = host->COMGETTER(NetworkInterfaces)(ComSafeArrayAsOutParam(nwInterfaces));
2035 if (FAILED(rc)) throw rc;
2036 /* We search for the first host network interface which
2037 * is usable for bridged networking */
2038 for (size_t i=0; i < nwInterfaces.size(); ++i)
2039 {
2040 HostNetworkInterfaceType_T itype;
2041 rc = nwInterfaces[i]->COMGETTER(InterfaceType)(&itype);
2042 if (FAILED(rc)) throw rc;
2043 if (itype == HostNetworkInterfaceType_Bridged)
2044 {
2045 Bstr name;
2046 rc = nwInterfaces[i]->COMGETTER(Name)(name.asOutParam());
2047 if (FAILED(rc)) throw rc;
2048 /* Set the interface name to attach to */
2049 pNetworkAdapter->COMSETTER(HostInterface)(name);
2050 if (FAILED(rc)) throw rc;
2051 break;
2052 }
2053 }
2054 }
2055 /* Next test for host only interfaces */
2056 else if (!pvsys->strExtraConfig.compare("type=HostOnly", Utf8Str::CaseInsensitive))
2057 {
2058 /* Attach to the right interface */
2059 rc = pNetworkAdapter->AttachToHostOnlyInterface();
2060 if (FAILED(rc)) throw rc;
2061 ComPtr<IHost> host;
2062 rc = pVirtualBox->COMGETTER(Host)(host.asOutParam());
2063 if (FAILED(rc)) throw rc;
2064 com::SafeIfaceArray<IHostNetworkInterface> nwInterfaces;
2065 rc = host->COMGETTER(NetworkInterfaces)(ComSafeArrayAsOutParam(nwInterfaces));
2066 if (FAILED(rc)) throw rc;
2067 /* We search for the first host network interface which
2068 * is usable for host only networking */
2069 for (size_t i=0; i < nwInterfaces.size(); ++i)
2070 {
2071 HostNetworkInterfaceType_T itype;
2072 rc = nwInterfaces[i]->COMGETTER(InterfaceType)(&itype);
2073 if (FAILED(rc)) throw rc;
2074 if (itype == HostNetworkInterfaceType_HostOnly)
2075 {
2076 Bstr name;
2077 rc = nwInterfaces[i]->COMGETTER(Name)(name.asOutParam());
2078 if (FAILED(rc)) throw rc;
2079 /* Set the interface name to attach to */
2080 pNetworkAdapter->COMSETTER(HostInterface)(name);
2081 if (FAILED(rc)) throw rc;
2082 break;
2083 }
2084 }
2085 }
2086 }
2087 }
2088
2089 /* Floppy drive */
2090 std::list<VirtualSystemDescriptionEntry*> vsdeFloppy = vsdescThis->findByType(VirtualSystemDescriptionType_Floppy);
2091 // Floppy support is enabled if there's at least one such entry; to disable floppy support,
2092 // the type of the floppy item would have been changed to "ignore"
2093 bool fFloppyEnabled = vsdeFloppy.size() > 0;
2094 ComPtr<IFloppyDrive> floppyDrive;
2095 rc = pNewMachine->COMGETTER(FloppyDrive)(floppyDrive.asOutParam());
2096 if (FAILED(rc)) throw rc;
2097 rc = floppyDrive->COMSETTER(Enabled)(fFloppyEnabled);
2098 if (FAILED(rc)) throw rc;
2099
2100 if (!task->progress.isNull())
2101 task->progress->notifyProgress((uint32_t)(opCountMax * opCount++));
2102
2103 /* CDROM drive */
2104 /* @todo: I can't disable the CDROM. So nothing to do for now */
2105 // std::list<VirtualSystemDescriptionEntry*> vsdeFloppy = vsd->findByType(VirtualSystemDescriptionType_CDROM);
2106
2107 /* Hard disk controller IDE */
2108 std::list<VirtualSystemDescriptionEntry*> vsdeHDCIDE = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskControllerIDE);
2109 if (vsdeHDCIDE.size() > 1)
2110 throw setError(VBOX_E_FILE_ERROR,
2111 tr("Too many IDE controllers in OVF; VirtualBox only supports one"));
2112 if (vsdeHDCIDE.size() == 1)
2113 {
2114 ComPtr<IStorageController> pController;
2115 rc = pNewMachine->GetStorageControllerByName(Bstr("IDE"), pController.asOutParam());
2116 if (FAILED(rc)) throw rc;
2117
2118 const char *pcszIDEType = vsdeHDCIDE.front()->strVbox.c_str();
2119 if (!strcmp(pcszIDEType, "PIIX3"))
2120 rc = pController->COMSETTER(ControllerType)(StorageControllerType_PIIX3);
2121 else if (!strcmp(pcszIDEType, "PIIX4"))
2122 rc = pController->COMSETTER(ControllerType)(StorageControllerType_PIIX4);
2123 else if (!strcmp(pcszIDEType, "ICH6"))
2124 rc = pController->COMSETTER(ControllerType)(StorageControllerType_ICH6);
2125 else
2126 throw setError(VBOX_E_FILE_ERROR,
2127 tr("Invalid IDE controller type \"%s\""),
2128 pcszIDEType);
2129 if (FAILED(rc)) throw rc;
2130 }
2131#ifdef VBOX_WITH_AHCI
2132 /* Hard disk controller SATA */
2133 std::list<VirtualSystemDescriptionEntry*> vsdeHDCSATA = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskControllerSATA);
2134 if (vsdeHDCSATA.size() > 1)
2135 throw setError(VBOX_E_FILE_ERROR,
2136 tr("Too many SATA controllers in OVF; VirtualBox only supports one"));
2137 if (vsdeHDCSATA.size() > 0)
2138 {
2139 ComPtr<IStorageController> pController;
2140 const Utf8Str &hdcVBox = vsdeHDCSATA.front()->strVbox;
2141 if (hdcVBox == "AHCI")
2142 {
2143 rc = pNewMachine->AddStorageController(Bstr("SATA"), StorageBus_SATA, pController.asOutParam());
2144 if (FAILED(rc)) throw rc;
2145 }
2146 else
2147 throw setError(VBOX_E_FILE_ERROR,
2148 tr("Invalid SATA controller type \"%s\""),
2149 hdcVBox.c_str());
2150 }
2151#endif /* VBOX_WITH_AHCI */
2152
2153 /* Hard disk controller SCSI */
2154 std::list<VirtualSystemDescriptionEntry*> vsdeHDCSCSI = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskControllerSCSI);
2155 if (vsdeHDCSCSI.size() > 1)
2156 throw setError(VBOX_E_FILE_ERROR,
2157 tr("Too many SCSI controllers in OVF; VirtualBox only supports one"));
2158 if (vsdeHDCSCSI.size() > 0)
2159 {
2160 ComPtr<IStorageController> pController;
2161 StorageControllerType_T controllerType;
2162 const Utf8Str &hdcVBox = vsdeHDCSCSI.front()->strVbox;
2163 if (hdcVBox == "LsiLogic")
2164 controllerType = StorageControllerType_LsiLogic;
2165 else if (hdcVBox == "BusLogic")
2166 controllerType = StorageControllerType_BusLogic;
2167 else
2168 throw setError(VBOX_E_FILE_ERROR,
2169 tr("Invalid SCSI controller type \"%s\""),
2170 hdcVBox.c_str());
2171
2172 rc = pNewMachine->AddStorageController(Bstr("SCSI"), StorageBus_SCSI, pController.asOutParam());
2173 if (FAILED(rc)) throw rc;
2174 rc = pController->COMSETTER(ControllerType)(controllerType);
2175 if (FAILED(rc)) throw rc;
2176 }
2177
2178 /* Now its time to register the machine before we add any hard disks */
2179 rc = pVirtualBox->RegisterMachine(pNewMachine);
2180 if (FAILED(rc)) throw rc;
2181
2182 Guid newMachineId;
2183 rc = pNewMachine->COMGETTER(Id)(newMachineId.asOutParam());
2184 if (FAILED(rc)) throw rc;
2185
2186 if (!task->progress.isNull())
2187 task->progress->notifyProgress((uint32_t)(opCountMax * opCount++));
2188
2189 // store new machine for roll-back in case of errors
2190 llMachinesRegistered.push_back(newMachineId);
2191
2192 /* Create the hard disks & connect them to the appropriate controllers. */
2193 std::list<VirtualSystemDescriptionEntry*> avsdeHDs = vsdescThis->findByType(VirtualSystemDescriptionType_HardDiskImage);
2194 if (avsdeHDs.size() > 0)
2195 {
2196 /* If in the next block an error occur we have to deregister
2197 the machine, so make an extra try/catch block. */
2198 ComPtr<IHardDisk> srcHdVBox;
2199 bool fSourceHdNeedsClosing = false;
2200
2201 try
2202 {
2203 /* In order to attach hard disks we need to open a session
2204 * for the new machine */
2205 rc = pVirtualBox->OpenSession(session, newMachineId);
2206 if (FAILED(rc)) throw rc;
2207 fSessionOpen = true;
2208
2209 /* The disk image has to be on the same place as the OVF file. So
2210 * strip the filename out of the full file path. */
2211 Utf8Str strSrcDir = stripFilename(pAppliance->m->strPath);
2212
2213 /* Iterate over all given disk images */
2214 list<VirtualSystemDescriptionEntry*>::const_iterator itHD;
2215 for (itHD = avsdeHDs.begin();
2216 itHD != avsdeHDs.end();
2217 ++itHD)
2218 {
2219 VirtualSystemDescriptionEntry *vsdeHD = *itHD;
2220
2221 const char *pcszDstFilePath = vsdeHD->strVbox.c_str();
2222 /* Check if the destination file exists already or the
2223 * destination path is empty. */
2224 if ( !(*pcszDstFilePath)
2225 || RTPathExists(pcszDstFilePath)
2226 )
2227 /* This isn't allowed */
2228 throw setError(VBOX_E_FILE_ERROR,
2229 tr("Destination file '%s' exists",
2230 pcszDstFilePath));
2231
2232 /* Find the disk from the OVF's disk list */
2233 DiskImagesMap::const_iterator itDiskImage = pAppliance->m->mapDisks.find(vsdeHD->strRef);
2234 /* vsdeHD->strRef contains the disk identifier (e.g. "vmdisk1"), which should exist
2235 in the virtual system's disks map under that ID and also in the global images map. */
2236 VirtualDisksMap::const_iterator itVirtualDisk = vsysThis.mapVirtualDisks.find(vsdeHD->strRef);
2237
2238 if ( itDiskImage == pAppliance->m->mapDisks.end()
2239 || itVirtualDisk == vsysThis.mapVirtualDisks.end()
2240 )
2241 throw setError(E_FAIL,
2242 tr("Internal inconsistency looking up disk images."));
2243
2244 const DiskImage &di = itDiskImage->second;
2245 const VirtualDisk &vd = itVirtualDisk->second;
2246
2247 /* Make sure all target directories exists */
2248 rc = VirtualBox::ensureFilePathExists(pcszDstFilePath);
2249 if (FAILED(rc))
2250 throw rc;
2251
2252 ComPtr<IProgress> progress;
2253
2254 ComPtr<IHardDisk> dstHdVBox;
2255 /* If strHref is empty we have to create a new file */
2256 if (di.strHref.isEmpty())
2257 {
2258 /* Which format to use? */
2259 Bstr srcFormat = L"VDI";
2260 if ( di.strFormat.compare("http://www.vmware.com/specifications/vmdk.html#sparse", Utf8Str::CaseInsensitive)
2261 || di.strFormat.compare("http://www.vmware.com/specifications/vmdk.html#compressed", Utf8Str::CaseInsensitive))
2262 srcFormat = L"VMDK";
2263 /* Create an empty hard disk */
2264 rc = pVirtualBox->CreateHardDisk(srcFormat, Bstr(pcszDstFilePath), dstHdVBox.asOutParam());
2265 if (FAILED(rc)) throw rc;
2266
2267 /* Create a dynamic growing disk image with the given capacity */
2268 rc = dstHdVBox->CreateBaseStorage(di.iCapacity / _1M, HardDiskVariant_Standard, progress.asOutParam());
2269 if (FAILED(rc)) throw rc;
2270
2271 /* Advance to the next operation */
2272 if (!task->progress.isNull())
2273 task->progress->advanceOperation (BstrFmt(tr("Creating virtual disk image '%s'"), pcszDstFilePath));
2274 }
2275 else
2276 {
2277 /* Construct the source file path */
2278 Utf8StrFmt strSrcFilePath("%s%c%s", strSrcDir.c_str(), RTPATH_DELIMITER, di.strHref.c_str());
2279 /* Check if the source file exists */
2280 if (!RTPathExists(strSrcFilePath.c_str()))
2281 /* This isn't allowed */
2282 throw setError(VBOX_E_FILE_ERROR,
2283 tr("Source virtual disk image file '%s' doesn't exist"),
2284 strSrcFilePath.c_str());
2285
2286 /* Clone the disk image (this is necessary cause the id has
2287 * to be recreated for the case the same hard disk is
2288 * attached already from a previous import) */
2289
2290 /* First open the existing disk image */
2291 rc = pVirtualBox->OpenHardDisk(Bstr(strSrcFilePath), srcHdVBox.asOutParam());
2292 if (FAILED(rc)) throw rc;
2293 fSourceHdNeedsClosing = true;
2294
2295 /* We need the format description of the source disk image */
2296 Bstr srcFormat;
2297 rc = srcHdVBox->COMGETTER(Format)(srcFormat.asOutParam());
2298 if (FAILED(rc)) throw rc;
2299 /* Create a new hard disk interface for the destination disk image */
2300 rc = pVirtualBox->CreateHardDisk(srcFormat, Bstr(pcszDstFilePath), dstHdVBox.asOutParam());
2301 if (FAILED(rc)) throw rc;
2302 /* Clone the source disk image */
2303 rc = srcHdVBox->CloneTo(dstHdVBox, HardDiskVariant_Standard, progress.asOutParam());
2304 if (FAILED(rc)) throw rc;
2305
2306 /* Advance to the next operation */
2307 if (!task->progress.isNull())
2308 task->progress->advanceOperation (BstrFmt(tr("Importing virtual disk image '%s'"), strSrcFilePath.c_str()));
2309 }
2310
2311 // now loop until the asynchronous operation completes and then
2312 // report its result
2313 BOOL fCompleted;
2314 LONG currentPercent;
2315 while (SUCCEEDED(progress->COMGETTER(Completed(&fCompleted))))
2316 {
2317 rc = progress->COMGETTER(Percent(&currentPercent));
2318 if (FAILED(rc)) throw rc;
2319 if (!task->progress.isNull())
2320 task->progress->notifyProgress(currentPercent);
2321 if (fCompleted)
2322 break;
2323 /* Make sure the loop is not too tight */
2324 rc = progress->WaitForCompletion(100);
2325 if (FAILED(rc)) throw rc;
2326 }
2327 // report result of asynchronous operation
2328 HRESULT vrc;
2329 rc = progress->COMGETTER(ResultCode)(&vrc);
2330 if (FAILED(rc)) throw rc;
2331
2332 // if the thread of the progress object has an error, then
2333 // retrieve the error info from there, or it'll be lost
2334 if (FAILED(vrc))
2335 {
2336 ProgressErrorInfo info(progress);
2337 Utf8Str str(info.getText());
2338 const char *pcsz = str.c_str();
2339 HRESULT rc2 = setError(vrc,
2340 pcsz);
2341 throw rc2;
2342 }
2343
2344 if (fSourceHdNeedsClosing)
2345 {
2346 rc = srcHdVBox->Close();
2347 if (FAILED(rc)) throw rc;
2348 fSourceHdNeedsClosing = false;
2349 }
2350
2351 llHardDisksCreated.push_back(dstHdVBox);
2352
2353 /* Now use the new uuid to attach the disk image to our new machine */
2354 ComPtr<IMachine> sMachine;
2355 rc = session->COMGETTER(Machine)(sMachine.asOutParam());
2356 if (FAILED(rc)) throw rc;
2357 Guid hdId;
2358 rc = dstHdVBox->COMGETTER(Id)(hdId.asOutParam());
2359 if (FAILED(rc)) throw rc;
2360
2361 /* For now we assume we have one controller of every type only */
2362 HardDiskController hdc = (*vsysThis.mapControllers.find(vd.idController)).second;
2363
2364 // this is for rollback later
2365 MyHardDiskAttachment mhda;
2366 mhda.uuid = newMachineId;
2367 mhda.pMachine = pNewMachine;
2368
2369 switch (hdc.system)
2370 {
2371 case HardDiskController::IDE:
2372 // For the IDE bus, the channel parameter can be either 0 or 1, to specify the primary
2373 // or secondary IDE controller, respectively. For the primary controller of the IDE bus,
2374 // the device number can be either 0 or 1, to specify the master or the slave device,
2375 // respectively. For the secondary IDE controller, the device number is always 1 because
2376 // the master device is reserved for the CD-ROM drive.
2377 mhda.controllerType = Bstr("IDE");
2378 switch (vd.ulAddressOnParent)
2379 {
2380 case 0: // interpret this as primary master
2381 mhda.lChannel = (long)0;
2382 mhda.lDevice = (long)0;
2383 break;
2384
2385 case 1: // interpret this as primary slave
2386 mhda.lChannel = (long)0;
2387 mhda.lDevice = (long)1;
2388 break;
2389
2390 case 2: // interpret this as secondary slave
2391 mhda.lChannel = (long)1;
2392 mhda.lDevice = (long)1;
2393 break;
2394
2395 default:
2396 throw setError(VBOX_E_NOT_SUPPORTED,
2397 tr("Invalid channel %RI16 specified; IDE controllers support only 0, 1 or 2"), vd.ulAddressOnParent);
2398 break;
2399 }
2400 break;
2401
2402 case HardDiskController::SATA:
2403 mhda.controllerType = Bstr("SATA");
2404 mhda.lChannel = (long)vd.ulAddressOnParent;
2405 mhda.lDevice = (long)0;
2406 break;
2407
2408 case HardDiskController::SCSI:
2409 mhda.controllerType = Bstr("SCSI");
2410 mhda.lChannel = (long)vd.ulAddressOnParent;
2411 mhda.lDevice = (long)0;
2412 break;
2413
2414 default: break;
2415 }
2416
2417 Log(("Attaching disk %s to channel %d on device %d\n", pcszDstFilePath, mhda.lChannel, mhda.lDevice));
2418
2419 rc = sMachine->AttachHardDisk(hdId,
2420 mhda.controllerType,
2421 mhda.lChannel,
2422 mhda.lDevice);
2423 if (FAILED(rc)) throw rc;
2424
2425 llHardDiskAttachments.push_back(mhda);
2426
2427 rc = sMachine->SaveSettings();
2428 if (FAILED(rc)) throw rc;
2429 } // end for (itHD = avsdeHDs.begin();
2430
2431 // only now that we're done with all disks, close the session
2432 rc = session->Close();
2433 if (FAILED(rc)) throw rc;
2434 fSessionOpen = false;
2435 }
2436 catch(HRESULT /* aRC */)
2437 {
2438 if (fSourceHdNeedsClosing)
2439 srcHdVBox->Close();
2440
2441 if (fSessionOpen)
2442 session->Close();
2443
2444 throw;
2445 }
2446 }
2447 }
2448 catch(HRESULT aRC)
2449 {
2450 rc = aRC;
2451 }
2452
2453 if (FAILED(rc))
2454 break;
2455
2456 } // for (it = pAppliance->m->llVirtualSystems.begin(),
2457
2458 if (FAILED(rc))
2459 {
2460 // with _whatever_ error we've had, do a complete roll-back of
2461 // machines and disks we've created; unfortunately this is
2462 // not so trivially done...
2463
2464 HRESULT rc2;
2465 // detach all hard disks from all machines we created
2466 list<MyHardDiskAttachment>::iterator itM;
2467 for (itM = llHardDiskAttachments.begin();
2468 itM != llHardDiskAttachments.end();
2469 ++itM)
2470 {
2471 const MyHardDiskAttachment &mhda = *itM;
2472 rc2 = pVirtualBox->OpenSession(session, mhda.uuid);
2473 if (SUCCEEDED(rc2))
2474 {
2475 ComPtr<IMachine> sMachine;
2476 rc2 = session->COMGETTER(Machine)(sMachine.asOutParam());
2477 if (SUCCEEDED(rc2))
2478 {
2479 rc2 = sMachine->DetachHardDisk(Bstr(mhda.controllerType), mhda.lChannel, mhda.lDevice);
2480 rc2 = sMachine->SaveSettings();
2481 }
2482 session->Close();
2483 }
2484 }
2485
2486 // now clean up all hard disks we created
2487 list< ComPtr<IHardDisk> >::iterator itHD;
2488 for (itHD = llHardDisksCreated.begin();
2489 itHD != llHardDisksCreated.end();
2490 ++itHD)
2491 {
2492 ComPtr<IHardDisk> pDisk = *itHD;
2493 ComPtr<IProgress> pProgress;
2494 rc2 = pDisk->DeleteStorage(pProgress.asOutParam());
2495 rc2 = pProgress->WaitForCompletion(-1);
2496 }
2497
2498 // finally, deregister and remove all machines
2499 list<Guid>::iterator itID;
2500 for (itID = llMachinesRegistered.begin();
2501 itID != llMachinesRegistered.end();
2502 ++itID)
2503 {
2504 const Guid &guid = *itID;
2505 ComPtr<IMachine> failedMachine;
2506 rc2 = pVirtualBox->UnregisterMachine(guid, failedMachine.asOutParam());
2507 if (SUCCEEDED(rc2))
2508 rc2 = failedMachine->DeleteSettings();
2509 }
2510 }
2511
2512 task->rc = rc;
2513
2514 if (!task->progress.isNull())
2515 task->progress->notifyComplete(rc);
2516
2517 LogFlowFunc(("rc=%Rhrc\n", rc));
2518 LogFlowFuncLeave();
2519
2520 return VINF_SUCCESS;
2521}
2522
2523/**
2524 * Worker thread implementation for Write() (ovf writer).
2525 * @param aThread
2526 * @param pvUser
2527 */
2528/* static */
2529DECLCALLBACK(int) Appliance::taskThreadWriteOVF(RTTHREAD /* aThread */, void *pvUser)
2530{
2531 std::auto_ptr<TaskWriteOVF> task(static_cast<TaskWriteOVF*>(pvUser));
2532 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
2533
2534 Appliance *pAppliance = task->pAppliance;
2535
2536 LogFlowFuncEnter();
2537 LogFlowFunc(("Appliance %p\n", pAppliance));
2538
2539 AutoCaller autoCaller(pAppliance);
2540 CheckComRCReturnRC(autoCaller.rc());
2541
2542 AutoWriteLock appLock(pAppliance);
2543
2544 HRESULT rc = S_OK;
2545
2546 ComPtr<IVirtualBox> pVirtualBox(pAppliance->mVirtualBox);
2547
2548 try
2549 {
2550 xml::Document doc;
2551 xml::ElementNode *pelmRoot = doc.createRootElement("Envelope");
2552
2553 pelmRoot->setAttribute("ovf:version", "1.0");
2554 pelmRoot->setAttribute("xml:lang", "en-US");
2555 pelmRoot->setAttribute("xmlns", "http://schemas.dmtf.org/ovf/envelope/1");
2556 pelmRoot->setAttribute("xmlns:ovf", "http://schemas.dmtf.org/ovf/envelope/1");
2557 pelmRoot->setAttribute("xmlns:ovfstr", "http://schema.dmtf.org/ovf/strings/1");
2558 pelmRoot->setAttribute("xmlns:rasd", "http://schemas.dmtf.org/wbem/wscim/1/cim-schema/2/CIM_ResourceAllocationSettingData");
2559 pelmRoot->setAttribute("xmlns:vssd", "http://schemas.dmtf.org/wbem/wscim/1/cim-schema/2/CIM_VirtualSystemSettingData");
2560 pelmRoot->setAttribute("xmlns:xsi", "http://www.w3.org/2001/XMLSchema-instance");
2561 pelmRoot->setAttribute("xsi:schemaLocation", "http://schemas.dmtf.org/ovf/envelope/1 ../ovf-envelope.xsd");
2562
2563
2564 // <Envelope>/<References>
2565 xml::ElementNode *pelmReferences = pelmRoot->createChild("References");
2566 // @ŧodo
2567
2568 /* <Envelope>/<DiskSection>:
2569 <DiskSection>
2570 <Info>List of the virtual disks used in the package</Info>
2571 <Disk ovf:capacity="4294967296" ovf:diskId="lamp" ovf:format="http://www.vmware.com/specifications/vmdk.html#compressed" ovf:populatedSize="1924967692"/>
2572 </DiskSection> */
2573 xml::ElementNode *pelmDiskSection = pelmRoot->createChild("DiskSection");
2574 xml::ElementNode *pelmDiskSectionInfo = pelmDiskSection->createChild("Info");
2575 pelmDiskSectionInfo->addContent("List of the virtual disks used in the package");
2576 // for now, set up a map so we have a list of unique disk names (to make
2577 // sure the same disk name is only added once)
2578 map<Utf8Str, const VirtualSystemDescriptionEntry*> mapDisks;
2579
2580 /* <Envelope>/<NetworkSection>:
2581 <NetworkSection>
2582 <Info>Logical networks used in the package</Info>
2583 <Network ovf:name="VM Network">
2584 <Description>The network that the LAMP Service will be available on</Description>
2585 </Network>
2586 </NetworkSection> */
2587 xml::ElementNode *pelmNetworkSection = pelmRoot->createChild("NetworkSection");
2588 xml::ElementNode *pelmNetworkSectionInfo = pelmNetworkSection->createChild("Info");
2589 pelmNetworkSectionInfo->addContent("Logical networks used in the package");
2590 // for now, set up a map so we have a list of unique network names (to make
2591 // sure the same network name is only added once)
2592 map<Utf8Str, bool> mapNetworks;
2593 // we fill this later below when we iterate over the networks
2594
2595 // and here come the virtual systems:
2596 xml::ElementNode *pelmVirtualSystemCollection = pelmRoot->createChild("VirtualSystemCollection");
2597 /* xml::AttributeNode *pattrVirtualSystemCollectionId = */ pelmVirtualSystemCollection->setAttribute("ovf:id", "ExportedVirtualBoxMachines"); // whatever
2598
2599 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it;
2600 /* Iterate through all virtual systems of that appliance */
2601 for (it = pAppliance->m->virtualSystemDescriptions.begin();
2602 it != pAppliance->m->virtualSystemDescriptions.end();
2603 ++it)
2604 {
2605 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it);
2606
2607 xml::ElementNode *pelmVirtualSystem = pelmVirtualSystemCollection->createChild("VirtualSystem");
2608
2609 xml::ElementNode *pelmVirtualSystemInfo = pelmVirtualSystem->createChild("Info");
2610 std::list<VirtualSystemDescriptionEntry*> llDescription = vsdescThis->findByType(VirtualSystemDescriptionType_Description);
2611 if (llDescription.size())
2612 pelmVirtualSystemInfo->addContent(llDescription.front()->strVbox);
2613
2614 std::list<VirtualSystemDescriptionEntry*> llName = vsdescThis->findByType(VirtualSystemDescriptionType_Name);
2615 if (llName.size() != 1)
2616 throw setError(VBOX_E_NOT_SUPPORTED,
2617 tr("Missing VM name"));
2618 pelmVirtualSystem->setAttribute("ovf:id", llName.front()->strVbox);
2619
2620 std::list<VirtualSystemDescriptionEntry*> llOS = vsdescThis->findByType(VirtualSystemDescriptionType_OS);
2621 if (llOS.size() != 1)
2622 throw setError(VBOX_E_NOT_SUPPORTED,
2623 tr("Missing OS type"));
2624 /* <OperatingSystemSection ovf:id="82">
2625 <Info>Guest Operating System</Info>
2626 <Description>Linux 2.6.x</Description>
2627 </OperatingSystemSection> */
2628 xml::ElementNode *pelmOperatingSystemSection = pelmVirtualSystem->createChild("OperatingSystemSection");
2629 pelmOperatingSystemSection->setAttribute("ovf:id", llOS.front()->strOvf);
2630// pelmOperatingSystemSection->createChild("Info")->addContent("blah"); // @todo
2631// pelmOperatingSystemSection->createChild("Description")->addContent("blah"); // @todo
2632
2633 // <VirtualHardwareSection ovf:id="hw1" ovf:transport="iso">
2634 xml::ElementNode *pelmVirtualHardwareSection = pelmVirtualSystem->createChild("VirtualHardwareSection");
2635
2636 /* <System>
2637 <vssd:Description>Description of the virtual hardware section.</vssd:Description>
2638 <vssd:ElementName>vmware</vssd:ElementName>
2639 <vssd:InstanceID>1</vssd:InstanceID>
2640 <vssd:VirtualSystemIdentifier>MyLampService</vssd:VirtualSystemIdentifier>
2641 <vssd:VirtualSystemType>vmx-4</vssd:VirtualSystemType>
2642 </System> */
2643 xml::ElementNode *pelmSystem = pelmVirtualHardwareSection->createChild("System");
2644
2645 // <vssd:VirtualSystemType>vmx-4</vssd:VirtualSystemType>
2646 xml::ElementNode *pelmVirtualSystemType = pelmSystem->createChild("VirtualSystemType");
2647 pelmVirtualSystemType->addContent("virtualbox-2.2"); // instead of vmx-7?
2648
2649 // loop thru all description entries twice; once to write out all
2650 // devices _except_ disk images, and a second time to assign the
2651 // disk images; this is because disk images need to reference
2652 // IDE controllers, and we can't know their instance IDs without
2653 // assigning them first
2654
2655 uint32_t idIDEController = 0;
2656 int32_t lIDEControllerIndex = 0;
2657 uint32_t idSATAController = 0;
2658 int32_t lSATAControllerIndex = 0;
2659 uint32_t idSCSIController = 0;
2660 int32_t lSCSIControllerIndex = 0;
2661
2662 uint32_t ulInstanceID = 1;
2663 uint32_t cDisks = 0;
2664
2665 for (size_t uLoop = 1;
2666 uLoop <= 2;
2667 ++uLoop)
2668 {
2669 int32_t lIndexThis = 0;
2670 list<VirtualSystemDescriptionEntry>::const_iterator itD;
2671 for (itD = vsdescThis->m->llDescriptions.begin();
2672 itD != vsdescThis->m->llDescriptions.end();
2673 ++itD, ++lIndexThis)
2674 {
2675 const VirtualSystemDescriptionEntry &desc = *itD;
2676
2677 OVFResourceType_T type = (OVFResourceType_T)0; // if this becomes != 0 then we do stuff
2678 Utf8Str strResourceSubType;
2679
2680 Utf8Str strDescription; // results in <rasd:Description>...</rasd:Description> block
2681 Utf8Str strCaption; // results in <rasd:Caption>...</rasd:Caption> block
2682
2683 uint32_t ulParent = 0;
2684
2685 int32_t lVirtualQuantity = -1;
2686 Utf8Str strAllocationUnits;
2687
2688 int32_t lAddress = -1;
2689 int32_t lBusNumber = -1;
2690 int32_t lAddressOnParent = -1;
2691
2692 int32_t lAutomaticAllocation = -1; // 0 means "false", 1 means "true"
2693 Utf8Str strConnection; // results in <rasd:Connection>...</rasd:Connection> block
2694 Utf8Str strHostResource;
2695
2696 uint64_t uTemp;
2697
2698 switch (desc.type)
2699 {
2700 case VirtualSystemDescriptionType_CPU:
2701 /* <Item>
2702 <rasd:Caption>1 virtual CPU</rasd:Caption>
2703 <rasd:Description>Number of virtual CPUs</rasd:Description>
2704 <rasd:ElementName>virtual CPU</rasd:ElementName>
2705 <rasd:InstanceID>1</rasd:InstanceID>
2706 <rasd:ResourceType>3</rasd:ResourceType>
2707 <rasd:VirtualQuantity>1</rasd:VirtualQuantity>
2708 </Item> */
2709 if (uLoop == 1)
2710 {
2711 strDescription = "Number of virtual CPUs";
2712 type = OVFResourceType_Processor; // 3
2713 lVirtualQuantity = 1;
2714 }
2715 break;
2716
2717 case VirtualSystemDescriptionType_Memory:
2718 /* <Item>
2719 <rasd:AllocationUnits>MegaBytes</rasd:AllocationUnits>
2720 <rasd:Caption>256 MB of memory</rasd:Caption>
2721 <rasd:Description>Memory Size</rasd:Description>
2722 <rasd:ElementName>Memory</rasd:ElementName>
2723 <rasd:InstanceID>2</rasd:InstanceID>
2724 <rasd:ResourceType>4</rasd:ResourceType>
2725 <rasd:VirtualQuantity>256</rasd:VirtualQuantity>
2726 </Item> */
2727 if (uLoop == 1)
2728 {
2729 strDescription = "Memory Size";
2730 type = OVFResourceType_Memory; // 4
2731 desc.strVbox.toInt(uTemp);
2732 lVirtualQuantity = (int32_t)(uTemp / _1M);
2733 strAllocationUnits = "MegaBytes";
2734 }
2735 break;
2736
2737 case VirtualSystemDescriptionType_HardDiskControllerIDE:
2738 /* <Item>
2739 <rasd:Caption>ideController1</rasd:Caption>
2740 <rasd:Description>IDE Controller</rasd:Description>
2741 <rasd:InstanceId>5</rasd:InstanceId>
2742 <rasd:ResourceType>5</rasd:ResourceType>
2743 <rasd:Address>1</rasd:Address>
2744 <rasd:BusNumber>1</rasd:BusNumber>
2745 </Item> */
2746 if (uLoop == 1)
2747 {
2748 strDescription = "IDE Controller";
2749 type = OVFResourceType_IDEController; // 5
2750 // it seems that OVFTool always writes these two, and since we can only
2751 // have one IDE controller, we'll use this as well
2752 lAddress = 1;
2753 lBusNumber = 1;
2754
2755 // remember this ID
2756 idIDEController = ulInstanceID;
2757 lIDEControllerIndex = lIndexThis;
2758 }
2759 break;
2760
2761 case VirtualSystemDescriptionType_HardDiskControllerSATA:
2762 /* <Item>
2763 <rasd:Caption>sataController0</rasd:Caption>
2764 <rasd:Description>SATA Controller</rasd:Description>
2765 <rasd:InstanceId>4</rasd:InstanceId>
2766 <rasd:ResourceType>20</rasd:ResourceType>
2767 <rasd:ResourceSubType>ahci</rasd:ResourceSubType>
2768 <rasd:Address>0</rasd:Address>
2769 <rasd:BusNumber>0</rasd:BusNumber>
2770 </Item>
2771 */
2772 if (uLoop == 1)
2773 {
2774 strDescription = "SATA Controller";
2775 strCaption = "sataController0";
2776 type = OVFResourceType_OtherStorageDevice; // 20
2777 // it seems that OVFTool always writes these two, and since we can only
2778 // have one SATA controller, we'll use this as well
2779 lAddress = 0;
2780 lBusNumber = 0;
2781
2782 if ( desc.strVbox.isEmpty() // AHCI is the default in VirtualBox
2783 || (!desc.strVbox.compare("ahci", Utf8Str::CaseInsensitive))
2784 )
2785 strResourceSubType = "AHCI";
2786 else
2787 throw setError(VBOX_E_NOT_SUPPORTED,
2788 tr("Invalid config string \"%s\" in SATA controller"), desc.strVbox.c_str());
2789
2790 // remember this ID
2791 idSATAController = ulInstanceID;
2792 lSATAControllerIndex = lIndexThis;
2793 }
2794 break;
2795
2796 case VirtualSystemDescriptionType_HardDiskControllerSCSI:
2797 /* <Item>
2798 <rasd:Caption>scsiController0</rasd:Caption>
2799 <rasd:Description>SCSI Controller</rasd:Description>
2800 <rasd:InstanceId>4</rasd:InstanceId>
2801 <rasd:ResourceType>6</rasd:ResourceType>
2802 <rasd:ResourceSubType>buslogic</rasd:ResourceSubType>
2803 <rasd:Address>0</rasd:Address>
2804 <rasd:BusNumber>0</rasd:BusNumber>
2805 </Item>
2806 */
2807 if (uLoop == 1)
2808 {
2809 strDescription = "SCSI Controller";
2810 strCaption = "scsiController0";
2811 type = OVFResourceType_ParallelSCSIHBA; // 6
2812 // it seems that OVFTool always writes these two, and since we can only
2813 // have one SATA controller, we'll use this as well
2814 lAddress = 0;
2815 lBusNumber = 0;
2816
2817 if ( desc.strVbox.isEmpty() // LsiLogic is the default in VirtualBox
2818 || (!desc.strVbox.compare("lsilogic", Utf8Str::CaseInsensitive))
2819 )
2820 strResourceSubType = "lsilogic";
2821 else if (!desc.strVbox.compare("buslogic", Utf8Str::CaseInsensitive))
2822 strResourceSubType = "buslogic";
2823 else
2824 throw setError(VBOX_E_NOT_SUPPORTED,
2825 tr("Invalid config string \"%s\" in SCSI controller"), desc.strVbox.c_str());
2826
2827 // remember this ID
2828 idSCSIController = ulInstanceID;
2829 lSCSIControllerIndex = lIndexThis;
2830 }
2831 break;
2832
2833 case VirtualSystemDescriptionType_HardDiskImage:
2834 /* <Item>
2835 <rasd:Caption>disk1</rasd:Caption>
2836 <rasd:InstanceId>8</rasd:InstanceId>
2837 <rasd:ResourceType>17</rasd:ResourceType>
2838 <rasd:HostResource>/disk/vmdisk1</rasd:HostResource>
2839 <rasd:Parent>4</rasd:Parent>
2840 <rasd:AddressOnParent>0</rasd:AddressOnParent>
2841 </Item> */
2842 if (uLoop == 2)
2843 {
2844 Utf8Str strDiskID = Utf8StrFmt("vmdisk%RI32", ++cDisks);
2845
2846 strDescription = "Disk Image";
2847 strCaption = Utf8StrFmt("disk%RI32", cDisks); // this is not used for anything else
2848 type = OVFResourceType_HardDisk; // 17
2849
2850 // the following references the "<Disks>" XML block
2851 strHostResource = Utf8StrFmt("/disk/%s", strDiskID.c_str());
2852
2853 // controller=<index>;channel=<c>
2854 size_t pos1 = desc.strExtraConfig.find("controller=");
2855 size_t pos2 = desc.strExtraConfig.find("channel=");
2856 if (pos1 != Utf8Str::npos)
2857 {
2858 int32_t lControllerIndex = -1;
2859 RTStrToInt32Ex(desc.strExtraConfig.c_str() + pos1 + 11, NULL, 0, &lControllerIndex);
2860 if (lControllerIndex == lIDEControllerIndex)
2861 ulParent = idIDEController;
2862 else if (lControllerIndex == lSCSIControllerIndex)
2863 ulParent = idSCSIController;
2864 else if (lControllerIndex == lSATAControllerIndex)
2865 ulParent = idSATAController;
2866 }
2867 if (pos2 != Utf8Str::npos)
2868 RTStrToInt32Ex(desc.strExtraConfig.c_str() + pos2 + 8, NULL, 0, &lAddressOnParent);
2869
2870 if ( !ulParent
2871 || lAddressOnParent == -1
2872 )
2873 throw setError(VBOX_E_NOT_SUPPORTED,
2874 tr("Missing or bad extra config string in hard disk image: \"%s\""), desc.strExtraConfig.c_str());
2875
2876 mapDisks[strDiskID] = &desc;
2877 }
2878 break;
2879
2880 case VirtualSystemDescriptionType_Floppy:
2881 if (uLoop == 1)
2882 {
2883 strDescription = "Floppy Drive";
2884 strCaption = "floppy0"; // this is what OVFTool writes
2885 type = OVFResourceType_FloppyDrive; // 14
2886 lAutomaticAllocation = 0;
2887 lAddressOnParent = 0; // this is what OVFTool writes
2888 }
2889 break;
2890
2891 case VirtualSystemDescriptionType_CDROM:
2892 if (uLoop == 2)
2893 {
2894 // we can't have a CD without an IDE controller
2895 if (!idIDEController)
2896 throw setError(VBOX_E_NOT_SUPPORTED,
2897 tr("Can't have CD-ROM without IDE controller"));
2898
2899 strDescription = "CD-ROM Drive";
2900 strCaption = "cdrom1"; // this is what OVFTool writes
2901 type = OVFResourceType_CDDrive; // 15
2902 lAutomaticAllocation = 1;
2903 ulParent = idIDEController;
2904 lAddressOnParent = 0; // this is what OVFTool writes
2905 }
2906 break;
2907
2908 case VirtualSystemDescriptionType_NetworkAdapter:
2909 /* <Item>
2910 <rasd:AutomaticAllocation>true</rasd:AutomaticAllocation>
2911 <rasd:Caption>Ethernet adapter on 'VM Network'</rasd:Caption>
2912 <rasd:Connection>VM Network</rasd:Connection>
2913 <rasd:ElementName>VM network</rasd:ElementName>
2914 <rasd:InstanceID>3</rasd:InstanceID>
2915 <rasd:ResourceType>10</rasd:ResourceType>
2916 </Item> */
2917 if (uLoop == 1)
2918 {
2919 lAutomaticAllocation = 1;
2920 strCaption = Utf8StrFmt("Ethernet adapter on '%s'", desc.strOvf.c_str());
2921 type = OVFResourceType_EthernetAdapter; // 10
2922 /* Set the hardware type to something useful.
2923 * To be compatible with vmware & others we set
2924 * PCNet32 for our PCNet types & E1000 for the
2925 * E1000 cards. */
2926 switch (desc.strVbox.toInt32())
2927 {
2928 case NetworkAdapterType_Am79C970A:
2929 case NetworkAdapterType_Am79C973: strResourceSubType = "PCNet32"; break;
2930#ifdef VBOX_WITH_E1000
2931 case NetworkAdapterType_I82540EM:
2932 case NetworkAdapterType_I82543GC: strResourceSubType = "E1000"; break;
2933#endif /* VBOX_WITH_E1000 */
2934 }
2935 strConnection = desc.strOvf;
2936
2937 mapNetworks[desc.strOvf] = true;
2938 }
2939 break;
2940
2941 case VirtualSystemDescriptionType_USBController:
2942 /* <Item ovf:required="false">
2943 <rasd:Caption>usb</rasd:Caption>
2944 <rasd:Description>USB Controller</rasd:Description>
2945 <rasd:InstanceId>3</rasd:InstanceId>
2946 <rasd:ResourceType>23</rasd:ResourceType>
2947 <rasd:Address>0</rasd:Address>
2948 <rasd:BusNumber>0</rasd:BusNumber>
2949 </Item> */
2950 if (uLoop == 1)
2951 {
2952 strDescription = "USB Controller";
2953 strCaption = "usb";
2954 type = OVFResourceType_USBController; // 23
2955 lAddress = 0; // this is what OVFTool writes
2956 lBusNumber = 0; // this is what OVFTool writes
2957 }
2958 break;
2959
2960 case VirtualSystemDescriptionType_SoundCard:
2961 /* <Item ovf:required="false">
2962 <rasd:Caption>sound</rasd:Caption>
2963 <rasd:Description>Sound Card</rasd:Description>
2964 <rasd:InstanceId>10</rasd:InstanceId>
2965 <rasd:ResourceType>35</rasd:ResourceType>
2966 <rasd:ResourceSubType>ensoniq1371</rasd:ResourceSubType>
2967 <rasd:AutomaticAllocation>false</rasd:AutomaticAllocation>
2968 <rasd:AddressOnParent>3</rasd:AddressOnParent>
2969 </Item> */
2970 if (uLoop == 1)
2971 {
2972 strDescription = "Sound Card";
2973 strCaption = "sound";
2974 type = OVFResourceType_SoundCard; // 35
2975 strResourceSubType = desc.strOvf; // e.g. ensoniq1371
2976 lAutomaticAllocation = 0;
2977 lAddressOnParent = 3; // what gives? this is what OVFTool writes
2978 }
2979 break;
2980 }
2981
2982 if (type)
2983 {
2984 xml::ElementNode *pItem;
2985
2986 pItem = pelmVirtualHardwareSection->createChild("Item");
2987
2988 if (!strDescription.isEmpty())
2989 pItem->createChild("rasd:Description")->addContent(strDescription);
2990 if (!strCaption.isEmpty())
2991 pItem->createChild("rasd:Caption")->addContent(strCaption);
2992
2993 if (!strAllocationUnits.isEmpty())
2994 pItem->createChild("rasd:AllocationUnits")->addContent(strAllocationUnits);
2995
2996 if (lAutomaticAllocation != -1)
2997 pItem->createChild("rasd:AutomaticAllocation")->addContent( (lAutomaticAllocation) ? "true" : "false" );
2998
2999 if (!strConnection.isEmpty())
3000 pItem->createChild("rasd:Connection")->addContent(strConnection);
3001
3002 // <rasd:InstanceID>1</rasd:InstanceID>
3003 pItem->createChild("rasd:InstanceID")->addContent(Utf8StrFmt("%d", ulInstanceID));
3004 ++ulInstanceID;
3005
3006 // <rasd:ResourceType>3</rasd:ResourceType>
3007 pItem->createChild("rasd:ResourceType")->addContent(Utf8StrFmt("%d", type));
3008 if (!strResourceSubType.isEmpty())
3009 pItem->createChild("rasd:ResourceSubType")->addContent(strResourceSubType);
3010
3011 // <rasd:VirtualQuantity>1</rasd:VirtualQuantity>
3012 if (lVirtualQuantity != -1)
3013 pItem->createChild("rasd:VirtualQuantity")->addContent(Utf8StrFmt("%d", lVirtualQuantity));
3014
3015 if (lAddress != -1)
3016 pItem->createChild("rasd:Address")->addContent(Utf8StrFmt("%d", lAddress));
3017
3018 if (lBusNumber != -1)
3019 pItem->createChild("rasd:BusNumber")->addContent(Utf8StrFmt("%d", lBusNumber));
3020
3021 if (ulParent)
3022 pItem->createChild("rasd:Parent")->addContent(Utf8StrFmt("%d", ulParent));
3023 if (lAddressOnParent != -1)
3024 pItem->createChild("rasd:AddressOnParent")->addContent(Utf8StrFmt("%d", lAddressOnParent));
3025
3026 if (!strHostResource.isEmpty())
3027 pItem->createChild("rasd:HostResource")->addContent(strHostResource);
3028 }
3029 }
3030 } // for (size_t uLoop = 0; ...
3031 }
3032
3033 // finally, fill in the network section we set up empty above according
3034 // to the networks we found with the hardware items
3035 map<Utf8Str, bool>::const_iterator itN;
3036 for (itN = mapNetworks.begin();
3037 itN != mapNetworks.end();
3038 ++itN)
3039 {
3040 const Utf8Str &strNetwork = itN->first;
3041 xml::ElementNode *pelmNetwork = pelmNetworkSection->createChild("Network");
3042 pelmNetwork->setAttribute("ovf:name", strNetwork.c_str());
3043 pelmNetwork->createChild("Description")->addContent("Logical network used by this appliance.");
3044 }
3045
3046 map<Utf8Str, const VirtualSystemDescriptionEntry*>::const_iterator itS;
3047 uint32_t ulFile = 1;
3048 for (itS = mapDisks.begin();
3049 itS != mapDisks.end();
3050 ++itS)
3051 {
3052 const Utf8Str &strDiskID = itS->first;
3053 const VirtualSystemDescriptionEntry *pDiskEntry = itS->second;
3054
3055 // source path: where the VBox image is
3056 const Utf8Str &strSrcFilePath = pDiskEntry->strVbox;
3057 Bstr bstrSrcFilePath(strSrcFilePath);
3058 if (!RTPathExists(strSrcFilePath.c_str()))
3059 /* This isn't allowed */
3060 throw setError(VBOX_E_FILE_ERROR,
3061 tr("Source virtual disk image file '%s' doesn't exist"),
3062 strSrcFilePath.c_str());
3063
3064 // output filename
3065 const Utf8Str &strTargetFileNameOnly = pDiskEntry->strOvf;
3066 // target path needs to be composed from where the output OVF is
3067 Utf8Str strTargetFilePath = stripFilename(pAppliance->m->strPath);
3068 strTargetFilePath.append("/");
3069 strTargetFilePath.append(strTargetFileNameOnly);
3070
3071 // clone the disk:
3072 ComPtr<IHardDisk> pSourceDisk;
3073 ComPtr<IHardDisk> pTargetDisk;
3074 ComPtr<IProgress> pProgress2;
3075
3076 Log(("Finding source disk \"%ls\"\n", bstrSrcFilePath.raw()));
3077 rc = pVirtualBox->FindHardDisk(bstrSrcFilePath, pSourceDisk.asOutParam());
3078 if (FAILED(rc)) throw rc;
3079
3080 /* We are always exporting to vmdfk stream optimized for now */
3081 Bstr bstrSrcFormat = L"VMDK";
3082
3083 // create a new hard disk interface for the destination disk image
3084 Log(("Creating target disk \"%s\"\n", strTargetFilePath.raw()));
3085 rc = pVirtualBox->CreateHardDisk(bstrSrcFormat, Bstr(strTargetFilePath), pTargetDisk.asOutParam());
3086 if (FAILED(rc)) throw rc;
3087
3088 // the target disk is now registered and needs to be removed again,
3089 // both after successful cloning or if anything goes bad!
3090 try
3091 {
3092 // clone the source disk image
3093 rc = pSourceDisk->CloneTo(pTargetDisk, HardDiskVariant_VmdkStreamOptimized, pProgress2.asOutParam());
3094 if (FAILED(rc)) throw rc;
3095
3096 // advance to the next operation
3097 if (!task->progress.isNull())
3098 task->progress->advanceOperation(BstrFmt(tr("Exporting virtual disk image '%s'"), strSrcFilePath.c_str()));
3099
3100 // now loop until the asynchronous operation completes and then
3101 // report its result
3102 BOOL fCompleted;
3103 LONG currentPercent;
3104 while (SUCCEEDED(pProgress2->COMGETTER(Completed(&fCompleted))))
3105 {
3106 rc = pProgress2->COMGETTER(Percent(&currentPercent));
3107 if (FAILED(rc)) throw rc;
3108 if (!task->progress.isNull())
3109 task->progress->notifyProgress(currentPercent);
3110 if (fCompleted)
3111 break;
3112 // make sure the loop is not too tight
3113 rc = pProgress2->WaitForCompletion(100);
3114 if (FAILED(rc)) throw rc;
3115 }
3116
3117 // report result of asynchronous operation
3118 HRESULT vrc;
3119 rc = pProgress2->COMGETTER(ResultCode)(&vrc);
3120 if (FAILED(rc)) throw rc;
3121
3122 // if the thread of the progress object has an error, then
3123 // retrieve the error info from there, or it'll be lost
3124 if (FAILED(vrc))
3125 {
3126 ProgressErrorInfo info(pProgress2);
3127 Utf8Str str(info.getText());
3128 const char *pcsz = str.c_str();
3129 HRESULT rc2 = setError(vrc, pcsz);
3130 throw rc2;
3131 }
3132 }
3133 catch (HRESULT rc3)
3134 {
3135 // upon error after registereing, close the disk or
3136 // it'll stick in the registry forever
3137 pTargetDisk->Close();
3138 throw rc3;
3139 }
3140
3141 // we need the following for the XML
3142 uint64_t cbFile = 0; // actual file size
3143 rc = pTargetDisk->COMGETTER(Size)(&cbFile);
3144 if (FAILED(rc)) throw rc;
3145
3146 ULONG64 cbCapacity = 0; // size reported to guest
3147 rc = pTargetDisk->COMGETTER(LogicalSize)(&cbCapacity);
3148 if (FAILED(rc)) throw rc;
3149 // capacity is reported in megabytes, so...
3150 cbCapacity *= _1M;
3151
3152 // upon success, close the disk as well
3153 rc = pTargetDisk->Close();
3154 if (FAILED(rc)) throw rc;
3155
3156 // now handle the XML for the disk:
3157 Utf8StrFmt strFileRef("file%RI32", ulFile++);
3158 // <File ovf:href="WindowsXpProfessional-disk1.vmdk" ovf:id="file1" ovf:size="1710381056"/>
3159 xml::ElementNode *pelmFile = pelmReferences->createChild("File");
3160 pelmFile->setAttribute("ovf:href", strTargetFileNameOnly);
3161 pelmFile->setAttribute("ovf:id", strFileRef);
3162 pelmFile->setAttribute("ovf:size", Utf8StrFmt("%RI64", cbFile).c_str());
3163
3164 // add disk to XML Disks section
3165 // <Disk ovf:capacity="8589934592" ovf:diskId="vmdisk1" ovf:fileRef="file1" ovf:format="http://www.vmware.com/specifications/vmdk.html#sparse"/>
3166 xml::ElementNode *pelmDisk = pelmDiskSection->createChild("Disk");
3167 pelmDisk->setAttribute("ovf:capacity", Utf8StrFmt("%RI64", cbCapacity).c_str());
3168 pelmDisk->setAttribute("ovf:diskId", strDiskID);
3169 pelmDisk->setAttribute("ovf:fileRef", strFileRef);
3170 pelmDisk->setAttribute("ovf:format", "http://www.vmware.com/specifications/vmdk.html#compressed");
3171 }
3172
3173 // now go write the XML
3174 xml::XmlFileWriter writer(doc);
3175 writer.write(pAppliance->m->strPath.c_str());
3176 }
3177 catch(xml::Error &x)
3178 {
3179 rc = setError(VBOX_E_FILE_ERROR,
3180 x.what());
3181 }
3182 catch(HRESULT aRC)
3183 {
3184 rc = aRC;
3185 }
3186
3187 task->rc = rc;
3188
3189 if (!task->progress.isNull())
3190 task->progress->notifyComplete(rc);
3191
3192 LogFlowFunc(("rc=%Rhrc\n", rc));
3193 LogFlowFuncLeave();
3194
3195 return VINF_SUCCESS;
3196}
3197
3198////////////////////////////////////////////////////////////////////////////////
3199//
3200// IVirtualSystemDescription constructor / destructor
3201//
3202////////////////////////////////////////////////////////////////////////////////
3203
3204DEFINE_EMPTY_CTOR_DTOR(VirtualSystemDescription)
3205struct shutup3 {};
3206
3207/**
3208 * COM initializer.
3209 * @return
3210 */
3211HRESULT VirtualSystemDescription::init()
3212{
3213 /* Enclose the state transition NotReady->InInit->Ready */
3214 AutoInitSpan autoInitSpan(this);
3215 AssertReturn(autoInitSpan.isOk(), E_FAIL);
3216
3217 /* Initialize data */
3218 m = new Data();
3219
3220 /* Confirm a successful initialization */
3221 autoInitSpan.setSucceeded();
3222 return S_OK;
3223}
3224
3225/**
3226* COM uninitializer.
3227*/
3228
3229void VirtualSystemDescription::uninit()
3230{
3231 delete m;
3232 m = NULL;
3233}
3234
3235////////////////////////////////////////////////////////////////////////////////
3236//
3237// IVirtualSystemDescription public methods
3238//
3239////////////////////////////////////////////////////////////////////////////////
3240
3241/**
3242 * Public method implementation.
3243 * @param
3244 * @return
3245 */
3246STDMETHODIMP VirtualSystemDescription::COMGETTER(Count)(ULONG *aCount)
3247{
3248 if (!aCount)
3249 return E_POINTER;
3250
3251 AutoCaller autoCaller(this);
3252 CheckComRCReturnRC(autoCaller.rc());
3253
3254 AutoReadLock alock(this);
3255
3256 *aCount = (ULONG)m->llDescriptions.size();
3257
3258 return S_OK;
3259}
3260
3261/**
3262 * Public method implementation.
3263 * @return
3264 */
3265STDMETHODIMP VirtualSystemDescription::GetDescription(ComSafeArrayOut(VirtualSystemDescriptionType_T, aTypes),
3266 ComSafeArrayOut(BSTR, aRefs),
3267 ComSafeArrayOut(BSTR, aOrigValues),
3268 ComSafeArrayOut(BSTR, aVboxValues),
3269 ComSafeArrayOut(BSTR, aExtraConfigValues))
3270{
3271 if (ComSafeArrayOutIsNull(aTypes) ||
3272 ComSafeArrayOutIsNull(aRefs) ||
3273 ComSafeArrayOutIsNull(aOrigValues) ||
3274 ComSafeArrayOutIsNull(aVboxValues) ||
3275 ComSafeArrayOutIsNull(aExtraConfigValues))
3276 return E_POINTER;
3277
3278 AutoCaller autoCaller(this);
3279 CheckComRCReturnRC(autoCaller.rc());
3280
3281 AutoReadLock alock(this);
3282
3283 ULONG c = (ULONG)m->llDescriptions.size();
3284 com::SafeArray<VirtualSystemDescriptionType_T> sfaTypes(c);
3285 com::SafeArray<BSTR> sfaRefs(c);
3286 com::SafeArray<BSTR> sfaOrigValues(c);
3287 com::SafeArray<BSTR> sfaVboxValues(c);
3288 com::SafeArray<BSTR> sfaExtraConfigValues(c);
3289
3290 list<VirtualSystemDescriptionEntry>::const_iterator it;
3291 size_t i = 0;
3292 for (it = m->llDescriptions.begin();
3293 it != m->llDescriptions.end();
3294 ++it, ++i)
3295 {
3296 const VirtualSystemDescriptionEntry &vsde = (*it);
3297
3298 sfaTypes[i] = vsde.type;
3299
3300 Bstr bstr = vsde.strRef;
3301 bstr.cloneTo(&sfaRefs[i]);
3302
3303 bstr = vsde.strOvf;
3304 bstr.cloneTo(&sfaOrigValues[i]);
3305
3306 bstr = vsde.strVbox;
3307 bstr.cloneTo(&sfaVboxValues[i]);
3308
3309 bstr = vsde.strExtraConfig;
3310 bstr.cloneTo(&sfaExtraConfigValues[i]);
3311 }
3312
3313 sfaTypes.detachTo(ComSafeArrayOutArg(aTypes));
3314 sfaRefs.detachTo(ComSafeArrayOutArg(aRefs));
3315 sfaOrigValues.detachTo(ComSafeArrayOutArg(aOrigValues));
3316 sfaVboxValues.detachTo(ComSafeArrayOutArg(aVboxValues));
3317 sfaExtraConfigValues.detachTo(ComSafeArrayOutArg(aExtraConfigValues));
3318
3319 return S_OK;
3320}
3321
3322/**
3323 * Public method implementation.
3324 * @return
3325 */
3326STDMETHODIMP VirtualSystemDescription::GetDescriptionByType(VirtualSystemDescriptionType_T aType,
3327 ComSafeArrayOut(VirtualSystemDescriptionType_T, aTypes),
3328 ComSafeArrayOut(BSTR, aRefs),
3329 ComSafeArrayOut(BSTR, aOrigValues),
3330 ComSafeArrayOut(BSTR, aVboxValues),
3331 ComSafeArrayOut(BSTR, aExtraConfigValues))
3332{
3333 if (ComSafeArrayOutIsNull(aTypes) ||
3334 ComSafeArrayOutIsNull(aRefs) ||
3335 ComSafeArrayOutIsNull(aOrigValues) ||
3336 ComSafeArrayOutIsNull(aVboxValues) ||
3337 ComSafeArrayOutIsNull(aExtraConfigValues))
3338 return E_POINTER;
3339
3340 AutoCaller autoCaller(this);
3341 CheckComRCReturnRC(autoCaller.rc());
3342
3343 AutoReadLock alock(this);
3344
3345 std::list<VirtualSystemDescriptionEntry*> vsd = findByType (aType);
3346 ULONG c = (ULONG)vsd.size();
3347 com::SafeArray<VirtualSystemDescriptionType_T> sfaTypes(c);
3348 com::SafeArray<BSTR> sfaRefs(c);
3349 com::SafeArray<BSTR> sfaOrigValues(c);
3350 com::SafeArray<BSTR> sfaVboxValues(c);
3351 com::SafeArray<BSTR> sfaExtraConfigValues(c);
3352
3353 list<VirtualSystemDescriptionEntry*>::const_iterator it;
3354 size_t i = 0;
3355 for (it = vsd.begin();
3356 it != vsd.end();
3357 ++it, ++i)
3358 {
3359 const VirtualSystemDescriptionEntry *vsde = (*it);
3360
3361 sfaTypes[i] = vsde->type;
3362
3363 Bstr bstr = vsde->strRef;
3364 bstr.cloneTo(&sfaRefs[i]);
3365
3366 bstr = vsde->strOvf;
3367 bstr.cloneTo(&sfaOrigValues[i]);
3368
3369 bstr = vsde->strVbox;
3370 bstr.cloneTo(&sfaVboxValues[i]);
3371
3372 bstr = vsde->strExtraConfig;
3373 bstr.cloneTo(&sfaExtraConfigValues[i]);
3374 }
3375
3376 sfaTypes.detachTo(ComSafeArrayOutArg(aTypes));
3377 sfaRefs.detachTo(ComSafeArrayOutArg(aRefs));
3378 sfaOrigValues.detachTo(ComSafeArrayOutArg(aOrigValues));
3379 sfaVboxValues.detachTo(ComSafeArrayOutArg(aVboxValues));
3380 sfaExtraConfigValues.detachTo(ComSafeArrayOutArg(aExtraConfigValues));
3381
3382 return S_OK;
3383}
3384
3385/**
3386 * Public method implementation.
3387 * @return
3388 */
3389STDMETHODIMP VirtualSystemDescription::SetFinalValues(ComSafeArrayIn(BOOL, aEnabled),
3390 ComSafeArrayIn(IN_BSTR, argVboxValues),
3391 ComSafeArrayIn(IN_BSTR, argExtraConfigValues))
3392{
3393#ifndef RT_OS_WINDOWS
3394 NOREF(aEnabledSize);
3395#endif /* RT_OS_WINDOWS */
3396
3397 CheckComArgSafeArrayNotNull(aEnabled);
3398 CheckComArgSafeArrayNotNull(argVboxValues);
3399 CheckComArgSafeArrayNotNull(argExtraConfigValues);
3400
3401 AutoCaller autoCaller(this);
3402 CheckComRCReturnRC(autoCaller.rc());
3403
3404 AutoWriteLock alock(this);
3405
3406 com::SafeArray<IN_BSTR> aVboxValues(ComSafeArrayInArg(argVboxValues));
3407 com::SafeArray<IN_BSTR> aExtraConfigValues(ComSafeArrayInArg(argExtraConfigValues));
3408
3409 if ( (aVboxValues.size() != m->llDescriptions.size())
3410 || (aExtraConfigValues.size() != m->llDescriptions.size())
3411 )
3412 return E_INVALIDARG;
3413
3414 list<VirtualSystemDescriptionEntry>::iterator it;
3415 size_t i = 0;
3416 for (it = m->llDescriptions.begin();
3417 it != m->llDescriptions.end();
3418 ++it, ++i)
3419 {
3420 VirtualSystemDescriptionEntry& vsde = *it;
3421
3422 if (aEnabled[i])
3423 {
3424 vsde.strVbox = aVboxValues[i];
3425 vsde.strExtraConfig = aExtraConfigValues[i];
3426 }
3427 else
3428 vsde.type = VirtualSystemDescriptionType_Ignore;
3429 }
3430
3431 return S_OK;
3432}
3433
3434/**
3435 * Internal method; adds a new description item to the member list.
3436 * @param aType Type of description for the new item.
3437 * @param strRef Reference item; only used with hard disk controllers.
3438 * @param aOrigValue Corresponding original value from OVF.
3439 * @param aAutoValue Initial configuration value (can be overridden by caller with setFinalValues).
3440 * @param strExtraConfig Extra configuration; meaning dependent on type.
3441 */
3442void VirtualSystemDescription::addEntry(VirtualSystemDescriptionType_T aType,
3443 const Utf8Str &strRef,
3444 const Utf8Str &aOrigValue,
3445 const Utf8Str &aAutoValue,
3446 const Utf8Str &strExtraConfig /*= ""*/)
3447{
3448 VirtualSystemDescriptionEntry vsde;
3449 vsde.ulIndex = (uint32_t)m->llDescriptions.size(); // each entry gets an index so the client side can reference them
3450 vsde.type = aType;
3451 vsde.strRef = strRef;
3452 vsde.strOvf = aOrigValue;
3453 vsde.strVbox = aAutoValue;
3454 vsde.strExtraConfig = strExtraConfig;
3455
3456 m->llDescriptions.push_back(vsde);
3457}
3458
3459/**
3460 * Private method; returns a list of description items containing all the items from the member
3461 * description items of this virtual system that match the given type.
3462 * @param aType
3463 * @return
3464 */
3465std::list<VirtualSystemDescriptionEntry*> VirtualSystemDescription::findByType(VirtualSystemDescriptionType_T aType)
3466{
3467 std::list<VirtualSystemDescriptionEntry*> vsd;
3468
3469 list<VirtualSystemDescriptionEntry>::iterator it;
3470 for (it = m->llDescriptions.begin();
3471 it != m->llDescriptions.end();
3472 ++it)
3473 {
3474 if (it->type == aType)
3475 vsd.push_back(&(*it));
3476 }
3477
3478 return vsd;
3479}
3480
3481/**
3482 * Private method; looks thru the member hardware items for the IDE, SATA, or SCSI controller with
3483 * the given reference ID. Useful when needing the controller for a particular
3484 * virtual disk.
3485 * @param id
3486 * @return
3487 */
3488const VirtualSystemDescriptionEntry* VirtualSystemDescription::findControllerFromID(uint32_t id)
3489{
3490 Utf8Str strRef = Utf8StrFmt("%RI32", id);
3491 list<VirtualSystemDescriptionEntry>::const_iterator it;
3492 for (it = m->llDescriptions.begin();
3493 it != m->llDescriptions.end();
3494 ++it)
3495 {
3496 const VirtualSystemDescriptionEntry &d = *it;
3497 switch (d.type)
3498 {
3499 case VirtualSystemDescriptionType_HardDiskControllerIDE:
3500 case VirtualSystemDescriptionType_HardDiskControllerSATA:
3501 case VirtualSystemDescriptionType_HardDiskControllerSCSI:
3502 if (d.strRef == strRef)
3503 return &d;
3504 break;
3505 }
3506 }
3507
3508 return NULL;
3509}
3510
3511////////////////////////////////////////////////////////////////////////////////
3512//
3513// IMachine public methods
3514//
3515////////////////////////////////////////////////////////////////////////////////
3516
3517// This code is here so we won't have to include the appliance headers in the
3518// IMachine implementation, and we also need to access private appliance data.
3519
3520/**
3521* Public method implementation.
3522* @param appliance
3523* @return
3524*/
3525
3526STDMETHODIMP Machine::Export(IAppliance *appliance)
3527{
3528 HRESULT rc = S_OK;
3529
3530 if (!appliance)
3531 return E_POINTER;
3532
3533 AutoCaller autoCaller(this);
3534 CheckComRCReturnRC(autoCaller.rc());
3535
3536 AutoReadLock alock(this);
3537
3538 ComObjPtr<VirtualSystemDescription> pNewDesc;
3539
3540 try
3541 {
3542 Bstr bstrName;
3543 Bstr bstrDescription;
3544 Bstr bstrGuestOSType;
3545 uint32_t cCPUs;
3546 uint32_t ulMemSizeMB;
3547 BOOL fDVDEnabled;
3548 BOOL fFloppyEnabled;
3549 BOOL fUSBEnabled;
3550 BOOL fAudioEnabled;
3551 AudioControllerType_T audioController;
3552
3553 ComPtr<IUSBController> pUsbController;
3554 ComPtr<IAudioAdapter> pAudioAdapter;
3555
3556 // get name
3557 bstrName = mUserData->mName;
3558 // get description
3559 bstrDescription = mUserData->mDescription;
3560 // get guest OS
3561 bstrGuestOSType = mUserData->mOSTypeId;
3562 // CPU count
3563 cCPUs = mHWData->mCPUCount;
3564 // memory size in MB
3565 ulMemSizeMB = mHWData->mMemorySize;
3566 // VRAM size?
3567 // BIOS settings?
3568 // 3D acceleration enabled?
3569 // hardware virtualization enabled?
3570 // nested paging enabled?
3571 // HWVirtExVPIDEnabled?
3572 // PAEEnabled?
3573 // snapshotFolder?
3574 // VRDPServer?
3575
3576 // floppy
3577 rc = mFloppyDrive->COMGETTER(Enabled)(&fFloppyEnabled);
3578 if (FAILED(rc)) throw rc;
3579
3580 // CD-ROM ?!?
3581 // ComPtr<IDVDDrive> pDVDDrive;
3582 fDVDEnabled = 1;
3583
3584 // this is more tricky so use the COM method
3585 rc = COMGETTER(USBController)(pUsbController.asOutParam());
3586 if (FAILED(rc)) throw rc;
3587 rc = pUsbController->COMGETTER(Enabled)(&fUSBEnabled);
3588
3589 pAudioAdapter = mAudioAdapter;
3590 rc = pAudioAdapter->COMGETTER(Enabled)(&fAudioEnabled);
3591 if (FAILED(rc)) throw rc;
3592 rc = pAudioAdapter->COMGETTER(AudioController)(&audioController);
3593 if (FAILED(rc)) throw rc;
3594
3595 // create a new virtual system
3596 rc = pNewDesc.createObject();
3597 CheckComRCThrowRC(rc);
3598 rc = pNewDesc->init();
3599 CheckComRCThrowRC(rc);
3600
3601 /* Guest OS type */
3602 Utf8Str strOsTypeVBox(bstrGuestOSType);
3603 CIMOSType_T cim = convertVBoxOSType2CIMOSType(strOsTypeVBox.c_str());
3604 pNewDesc->addEntry(VirtualSystemDescriptionType_OS,
3605 "",
3606 Utf8StrFmt("%RI32", cim),
3607 strOsTypeVBox);
3608
3609 /* VM name */
3610 Utf8Str strVMName(bstrName);
3611 pNewDesc->addEntry(VirtualSystemDescriptionType_Name,
3612 "",
3613 strVMName,
3614 strVMName);
3615
3616 // description
3617 Utf8Str strDescription(bstrDescription);
3618 pNewDesc->addEntry(VirtualSystemDescriptionType_Description,
3619 "",
3620 strDescription,
3621 strDescription);
3622
3623 /* CPU count*/
3624 Utf8Str strCpuCount = Utf8StrFmt("%RI32", cCPUs);
3625 pNewDesc->addEntry(VirtualSystemDescriptionType_CPU,
3626 "",
3627 strCpuCount,
3628 strCpuCount);
3629
3630 /* Memory */
3631 Utf8Str strMemory = Utf8StrFmt("%RI32", (uint64_t)ulMemSizeMB * _1M);
3632 pNewDesc->addEntry(VirtualSystemDescriptionType_Memory,
3633 "",
3634 strMemory,
3635 strMemory);
3636
3637 int32_t lIDEControllerIndex = 0;
3638 int32_t lSATAControllerIndex = 0;
3639 int32_t lSCSIControllerIndex = 0;
3640
3641// <const name="HardDiskControllerIDE" value="6" />
3642 ComPtr<IStorageController> pController;
3643 rc = GetStorageControllerByName(Bstr("IDE"), pController.asOutParam());
3644 if (FAILED(rc)) throw rc;
3645 Utf8Str strVbox;
3646 StorageControllerType_T ctlr;
3647 rc = pController->COMGETTER(ControllerType)(&ctlr);
3648 if (FAILED(rc)) throw rc;
3649 switch(ctlr)
3650 {
3651 case StorageControllerType_PIIX3: strVbox = "PIIX3"; break;
3652 case StorageControllerType_PIIX4: strVbox = "PIIX4"; break;
3653 case StorageControllerType_ICH6: strVbox = "ICH6"; break;
3654 }
3655
3656 if (strVbox.length())
3657 {
3658 lIDEControllerIndex = (int32_t)pNewDesc->m->llDescriptions.size();
3659 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerIDE,
3660 Utf8StrFmt("%d", lIDEControllerIndex),
3661 strVbox,
3662 strVbox);
3663 }
3664
3665#ifdef VBOX_WITH_AHCI
3666// <const name="HardDiskControllerSATA" value="7" />
3667 rc = GetStorageControllerByName(Bstr("SATA"), pController.asOutParam());
3668 if (SUCCEEDED(rc))
3669 {
3670 strVbox = "AHCI";
3671 lSATAControllerIndex = (int32_t)pNewDesc->m->llDescriptions.size();
3672 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerSATA,
3673 Utf8StrFmt("%d", lSATAControllerIndex),
3674 strVbox,
3675 strVbox);
3676 }
3677#endif // VBOX_WITH_AHCI
3678
3679// <const name="HardDiskControllerSCSI" value="8" />
3680 rc = GetStorageControllerByName(Bstr("SCSI"), pController.asOutParam());
3681 if (SUCCEEDED(rc))
3682 {
3683 rc = pController->COMGETTER(ControllerType)(&ctlr);
3684 if (SUCCEEDED(rc))
3685 {
3686 strVbox = "LsiLogic"; // the default in VBox
3687 switch(ctlr)
3688 {
3689 case StorageControllerType_LsiLogic: strVbox = "LsiLogic"; break;
3690 case StorageControllerType_BusLogic: strVbox = "BusLogic"; break;
3691 }
3692 lSCSIControllerIndex = (int32_t)pNewDesc->m->llDescriptions.size();
3693 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskControllerSCSI,
3694 Utf8StrFmt("%d", lSCSIControllerIndex),
3695 strVbox,
3696 strVbox);
3697 }
3698 else
3699 throw rc;
3700 }
3701
3702// <const name="HardDiskImage" value="9" />
3703 HDData::AttachmentList::iterator itA;
3704 for (itA = mHDData->mAttachments.begin();
3705 itA != mHDData->mAttachments.end();
3706 ++itA)
3707 {
3708 ComObjPtr<HardDiskAttachment> pHDA = *itA;
3709
3710 // the attachment's data
3711 ComPtr<IHardDisk> pHardDisk;
3712 ComPtr<IStorageController> ctl;
3713 Bstr controllerName;
3714
3715 rc = pHDA->COMGETTER(Controller)(controllerName.asOutParam());
3716 if (FAILED(rc)) throw rc;
3717
3718 rc = GetStorageControllerByName(controllerName, ctl.asOutParam());
3719 if (FAILED(rc)) throw rc;
3720
3721 StorageBus_T storageBus;
3722 LONG lChannel;
3723 LONG lDevice;
3724
3725 rc = ctl->COMGETTER(Bus)(&storageBus);
3726 if (FAILED(rc)) throw rc;
3727
3728 rc = pHDA->COMGETTER(HardDisk)(pHardDisk.asOutParam());
3729 if (FAILED(rc)) throw rc;
3730
3731 rc = pHDA->COMGETTER(Port)(&lChannel);
3732 if (FAILED(rc)) throw rc;
3733
3734 rc = pHDA->COMGETTER(Device)(&lDevice);
3735 if (FAILED(rc)) throw rc;
3736
3737 Bstr bstrLocation;
3738 rc = pHardDisk->COMGETTER(Location)(bstrLocation.asOutParam());
3739 Bstr bstrName;
3740 rc = pHardDisk->COMGETTER(Name)(bstrName.asOutParam());
3741
3742 // and how this translates to the virtual system
3743 int32_t lControllerVsys = 0;
3744 LONG lChannelVsys;
3745
3746 switch (storageBus)
3747 {
3748 case StorageBus_IDE:
3749 // this is the exact reverse to what we're doing in Appliance::taskThreadImportMachines,
3750 // and it must be updated when that is changed!
3751
3752 if (lChannel == 0 && lDevice == 0) // primary master
3753 lChannelVsys = 0;
3754 else if (lChannel == 0 && lDevice == 1) // primary slave
3755 lChannelVsys = 1;
3756 else if (lChannel == 1 && lDevice == 1) // secondary slave; secondary master is always CDROM
3757 lChannelVsys = 2;
3758 else
3759 throw setError(VBOX_E_NOT_SUPPORTED,
3760 tr("Cannot handle hard disk attachment: channel is %d, device is %d"), lChannel, lDevice);
3761
3762 lControllerVsys = lIDEControllerIndex;
3763 break;
3764
3765 case StorageBus_SATA:
3766 lChannelVsys = lChannel; // should be between 0 and 29
3767 lControllerVsys = lSATAControllerIndex;
3768 break;
3769
3770 case StorageBus_SCSI:
3771 lChannelVsys = lChannel; // should be between 0 and 15
3772 lControllerVsys = lSCSIControllerIndex;
3773 break;
3774
3775 default:
3776 throw setError(VBOX_E_NOT_SUPPORTED,
3777 tr("Cannot handle hard disk attachment: storageBus is %d, channel is %d, device is %d"), storageBus, lChannel, lDevice);
3778 break;
3779 }
3780
3781 Utf8Str strTargetVmdkName(bstrName);
3782 RTPathStripExt(strTargetVmdkName.mutableRaw());
3783 strTargetVmdkName.append(".vmdk");
3784
3785 pNewDesc->addEntry(VirtualSystemDescriptionType_HardDiskImage,
3786 strTargetVmdkName, // disk ID: let's use the name
3787 strTargetVmdkName, // OVF value:
3788 Utf8Str(bstrLocation), // vbox value: media path
3789 Utf8StrFmt("controller=%RI32;channel=%RI32", lControllerVsys, lChannelVsys));
3790 }
3791
3792 /* Floppy Drive */
3793 if (fFloppyEnabled)
3794 pNewDesc->addEntry(VirtualSystemDescriptionType_Floppy, "", "", "");
3795
3796 /* CD Drive */
3797 if (fDVDEnabled)
3798 pNewDesc->addEntry(VirtualSystemDescriptionType_CDROM, "", "", "");
3799
3800// <const name="NetworkAdapter" />
3801 size_t a;
3802 for (a = 0;
3803 a < SchemaDefs::NetworkAdapterCount;
3804 ++a)
3805 {
3806 ComPtr<INetworkAdapter> pNetworkAdapter;
3807 BOOL fEnabled;
3808 NetworkAdapterType_T adapterType;
3809 NetworkAttachmentType_T attachmentType;
3810
3811 rc = GetNetworkAdapter((ULONG)a, pNetworkAdapter.asOutParam());
3812 if (FAILED(rc)) throw rc;
3813 /* Enable the network card & set the adapter type */
3814 rc = pNetworkAdapter->COMGETTER(Enabled)(&fEnabled);
3815 if (FAILED(rc)) throw rc;
3816
3817 if (fEnabled)
3818 {
3819 Utf8Str strAttachmentType;
3820
3821 rc = pNetworkAdapter->COMGETTER(AdapterType)(&adapterType);
3822 if (FAILED(rc)) throw rc;
3823
3824 rc = pNetworkAdapter->COMGETTER(AttachmentType)(&attachmentType);
3825 if (FAILED(rc)) throw rc;
3826
3827 switch (attachmentType)
3828 {
3829 case NetworkAttachmentType_Null:
3830 strAttachmentType = "Null";
3831 break;
3832
3833 case NetworkAttachmentType_NAT:
3834 strAttachmentType = "NAT";
3835 break;
3836
3837 case NetworkAttachmentType_Bridged:
3838 strAttachmentType = "Bridged";
3839 break;
3840
3841 case NetworkAttachmentType_Internal:
3842 strAttachmentType = "Internal";
3843 break;
3844
3845 case NetworkAttachmentType_HostOnly:
3846 strAttachmentType = "HostOnly";
3847 break;
3848 }
3849
3850 pNewDesc->addEntry(VirtualSystemDescriptionType_NetworkAdapter,
3851 "", // ref
3852 strAttachmentType, // orig
3853 Utf8StrFmt("%RI32", (uint32_t)adapterType), // conf
3854 Utf8StrFmt("type=%s", strAttachmentType.c_str())); // extra conf
3855 }
3856 }
3857
3858// <const name="USBController" />
3859#ifdef VBOX_WITH_USB
3860 if (fUSBEnabled)
3861 pNewDesc->addEntry(VirtualSystemDescriptionType_USBController, "", "", "");
3862#endif /* VBOX_WITH_USB */
3863
3864// <const name="SoundCard" />
3865 if (fAudioEnabled)
3866 {
3867 pNewDesc->addEntry(VirtualSystemDescriptionType_SoundCard,
3868 "",
3869 "ensoniq1371", // this is what OVFTool writes and VMware supports
3870 Utf8StrFmt("%RI32", audioController));
3871 }
3872
3873 // finally, add the virtual system to the appliance
3874 Appliance *pAppliance = static_cast<Appliance*>(appliance);
3875 AutoCaller autoCaller(pAppliance);
3876 if (FAILED(rc)) throw rc;
3877
3878 AutoWriteLock alock(pAppliance);
3879
3880 pAppliance->m->virtualSystemDescriptions.push_back(pNewDesc);
3881 }
3882 catch(HRESULT arc)
3883 {
3884 rc = arc;
3885 }
3886
3887 return rc;
3888}
3889
3890/* 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