VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/ApplianceImplImport.cpp@ 59967

Last change on this file since 59967 was 59760, checked in by vboxsync, 9 years ago

Main: shadowed variable

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 174.5 KB
Line 
1/* $Id: ApplianceImplImport.cpp 59760 2016-02-20 16:50:04Z vboxsync $ */
2/** @file
3 * IAppliance and IVirtualSystem COM class implementations.
4 */
5
6/*
7 * Copyright (C) 2008-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18#include <iprt/alloca.h>
19#include <iprt/path.h>
20#include <iprt/dir.h>
21#include <iprt/file.h>
22#include <iprt/s3.h>
23#include <iprt/sha.h>
24#include <iprt/manifest.h>
25#include <iprt/tar.h>
26#include <iprt/zip.h>
27#include <iprt/stream.h>
28#include <iprt/crypto/digest.h>
29#include <iprt/crypto/pkix.h>
30#include <iprt/crypto/store.h>
31#include <iprt/crypto/x509.h>
32
33#include <VBox/vd.h>
34#include <VBox/com/array.h>
35
36#include "ApplianceImpl.h"
37#include "VirtualBoxImpl.h"
38#include "GuestOSTypeImpl.h"
39#include "ProgressImpl.h"
40#include "MachineImpl.h"
41#include "MediumImpl.h"
42#include "MediumFormatImpl.h"
43#include "SystemPropertiesImpl.h"
44#include "HostImpl.h"
45
46#include "AutoCaller.h"
47#include "Logging.h"
48
49#include "ApplianceImplPrivate.h"
50
51#include <VBox/param.h>
52#include <VBox/version.h>
53#include <VBox/settings.h>
54
55#include <set>
56
57using namespace std;
58
59////////////////////////////////////////////////////////////////////////////////
60//
61// IAppliance public methods
62//
63////////////////////////////////////////////////////////////////////////////////
64
65/**
66 * Public method implementation. This opens the OVF with ovfreader.cpp.
67 * Thread implementation is in Appliance::readImpl().
68 *
69 * @param aFile
70 * @return
71 */
72HRESULT Appliance::read(const com::Utf8Str &aFile,
73 ComPtr<IProgress> &aProgress)
74{
75 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
76
77 if (!i_isApplianceIdle())
78 return E_ACCESSDENIED;
79
80 if (m->pReader)
81 {
82 delete m->pReader;
83 m->pReader = NULL;
84 }
85
86 // see if we can handle this file; for now we insist it has an ovf/ova extension
87 if ( !aFile.endsWith(".ovf", Utf8Str::CaseInsensitive)
88 && !aFile.endsWith(".ova", Utf8Str::CaseInsensitive))
89 return setError(VBOX_E_FILE_ERROR, tr("Appliance file must have .ovf or .ova extension"));
90
91 ComObjPtr<Progress> progress;
92 try
93 {
94 /* Parse all necessary info out of the URI */
95 i_parseURI(aFile, m->locInfo);
96 i_readImpl(m->locInfo, progress);
97 }
98 catch (HRESULT aRC)
99 {
100 return aRC;
101 }
102
103 /* Return progress to the caller */
104 progress.queryInterfaceTo(aProgress.asOutParam());
105 return S_OK;
106}
107
108/**
109 * Public method implementation. This looks at the output of ovfreader.cpp and creates
110 * VirtualSystemDescription instances.
111 * @return
112 */
113HRESULT Appliance::interpret()
114{
115 // @todo:
116 // - don't use COM methods but the methods directly (faster, but needs appropriate
117 // locking of that objects itself (s. HardDisk))
118 // - Appropriate handle errors like not supported file formats
119 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
120
121 if (!i_isApplianceIdle())
122 return E_ACCESSDENIED;
123
124 HRESULT rc = S_OK;
125
126 /* Clear any previous virtual system descriptions */
127 m->virtualSystemDescriptions.clear();
128
129 if (!m->pReader)
130 return setError(E_FAIL,
131 tr("Cannot interpret appliance without reading it first (call read() before interpret())"));
132
133 // Change the appliance state so we can safely leave the lock while doing time-consuming
134 // disk imports; also the below method calls do all kinds of locking which conflicts with
135 // the appliance object lock
136 m->state = Data::ApplianceImporting;
137 alock.release();
138
139 /* Try/catch so we can clean up on error */
140 try
141 {
142 list<ovf::VirtualSystem>::const_iterator it;
143 /* Iterate through all virtual systems */
144 for (it = m->pReader->m_llVirtualSystems.begin();
145 it != m->pReader->m_llVirtualSystems.end();
146 ++it)
147 {
148 const ovf::VirtualSystem &vsysThis = *it;
149
150 ComObjPtr<VirtualSystemDescription> pNewDesc;
151 rc = pNewDesc.createObject();
152 if (FAILED(rc)) throw rc;
153 rc = pNewDesc->init();
154 if (FAILED(rc)) throw rc;
155
156 // if the virtual system in OVF had a <vbox:Machine> element, have the
157 // VirtualBox settings code parse that XML now
158 if (vsysThis.pelmVBoxMachine)
159 pNewDesc->i_importVBoxMachineXML(*vsysThis.pelmVBoxMachine);
160
161 // Guest OS type
162 // This is taken from one of three places, in this order:
163 Utf8Str strOsTypeVBox;
164 Utf8StrFmt strCIMOSType("%RU32", (uint32_t)vsysThis.cimos);
165 // 1) If there is a <vbox:Machine>, then use the type from there.
166 if ( vsysThis.pelmVBoxMachine
167 && pNewDesc->m->pConfig->machineUserData.strOsType.isNotEmpty()
168 )
169 strOsTypeVBox = pNewDesc->m->pConfig->machineUserData.strOsType;
170 // 2) Otherwise, if there is OperatingSystemSection/vbox:OSType, use that one.
171 else if (vsysThis.strTypeVBox.isNotEmpty()) // OVFReader has found vbox:OSType
172 strOsTypeVBox = vsysThis.strTypeVBox;
173 // 3) Otherwise, make a best guess what the vbox type is from the OVF (CIM) OS type.
174 else
175 convertCIMOSType2VBoxOSType(strOsTypeVBox, vsysThis.cimos, vsysThis.strCimosDesc);
176 pNewDesc->i_addEntry(VirtualSystemDescriptionType_OS,
177 "",
178 strCIMOSType,
179 strOsTypeVBox);
180
181 /* VM name */
182 Utf8Str nameVBox;
183 /* If there is a <vbox:Machine>, we always prefer the setting from there. */
184 if ( vsysThis.pelmVBoxMachine
185 && pNewDesc->m->pConfig->machineUserData.strName.isNotEmpty())
186 nameVBox = pNewDesc->m->pConfig->machineUserData.strName;
187 else
188 nameVBox = vsysThis.strName;
189 /* If there isn't any name specified create a default one out
190 * of the OS type */
191 if (nameVBox.isEmpty())
192 nameVBox = strOsTypeVBox;
193 i_searchUniqueVMName(nameVBox);
194 pNewDesc->i_addEntry(VirtualSystemDescriptionType_Name,
195 "",
196 vsysThis.strName,
197 nameVBox);
198
199 /* Based on the VM name, create a target machine path. */
200 Bstr bstrMachineFilename;
201 rc = mVirtualBox->ComposeMachineFilename(Bstr(nameVBox).raw(),
202 NULL /* aGroup */,
203 NULL /* aCreateFlags */,
204 NULL /* aBaseFolder */,
205 bstrMachineFilename.asOutParam());
206 if (FAILED(rc)) throw rc;
207 /* Determine the machine folder from that */
208 Utf8Str strMachineFolder = Utf8Str(bstrMachineFilename).stripFilename();
209
210 /* VM Product */
211 if (!vsysThis.strProduct.isEmpty())
212 pNewDesc->i_addEntry(VirtualSystemDescriptionType_Product,
213 "",
214 vsysThis.strProduct,
215 vsysThis.strProduct);
216
217 /* VM Vendor */
218 if (!vsysThis.strVendor.isEmpty())
219 pNewDesc->i_addEntry(VirtualSystemDescriptionType_Vendor,
220 "",
221 vsysThis.strVendor,
222 vsysThis.strVendor);
223
224 /* VM Version */
225 if (!vsysThis.strVersion.isEmpty())
226 pNewDesc->i_addEntry(VirtualSystemDescriptionType_Version,
227 "",
228 vsysThis.strVersion,
229 vsysThis.strVersion);
230
231 /* VM ProductUrl */
232 if (!vsysThis.strProductUrl.isEmpty())
233 pNewDesc->i_addEntry(VirtualSystemDescriptionType_ProductUrl,
234 "",
235 vsysThis.strProductUrl,
236 vsysThis.strProductUrl);
237
238 /* VM VendorUrl */
239 if (!vsysThis.strVendorUrl.isEmpty())
240 pNewDesc->i_addEntry(VirtualSystemDescriptionType_VendorUrl,
241 "",
242 vsysThis.strVendorUrl,
243 vsysThis.strVendorUrl);
244
245 /* VM description */
246 if (!vsysThis.strDescription.isEmpty())
247 pNewDesc->i_addEntry(VirtualSystemDescriptionType_Description,
248 "",
249 vsysThis.strDescription,
250 vsysThis.strDescription);
251
252 /* VM license */
253 if (!vsysThis.strLicenseText.isEmpty())
254 pNewDesc->i_addEntry(VirtualSystemDescriptionType_License,
255 "",
256 vsysThis.strLicenseText,
257 vsysThis.strLicenseText);
258
259 /* Now that we know the OS type, get our internal defaults based on that. */
260 ComPtr<IGuestOSType> pGuestOSType;
261 rc = mVirtualBox->GetGuestOSType(Bstr(strOsTypeVBox).raw(), pGuestOSType.asOutParam());
262 if (FAILED(rc)) throw rc;
263
264 /* CPU count */
265 ULONG cpuCountVBox;
266 /* If there is a <vbox:Machine>, we always prefer the setting from there. */
267 if ( vsysThis.pelmVBoxMachine
268 && pNewDesc->m->pConfig->hardwareMachine.cCPUs)
269 cpuCountVBox = pNewDesc->m->pConfig->hardwareMachine.cCPUs;
270 else
271 cpuCountVBox = vsysThis.cCPUs;
272 /* Check for the constraints */
273 if (cpuCountVBox > SchemaDefs::MaxCPUCount)
274 {
275 i_addWarning(tr("The virtual system \"%s\" claims support for %u CPU's, but VirtualBox has support for "
276 "max %u CPU's only."),
277 vsysThis.strName.c_str(), cpuCountVBox, SchemaDefs::MaxCPUCount);
278 cpuCountVBox = SchemaDefs::MaxCPUCount;
279 }
280 if (vsysThis.cCPUs == 0)
281 cpuCountVBox = 1;
282 pNewDesc->i_addEntry(VirtualSystemDescriptionType_CPU,
283 "",
284 Utf8StrFmt("%RU32", (uint32_t)vsysThis.cCPUs),
285 Utf8StrFmt("%RU32", (uint32_t)cpuCountVBox));
286
287 /* RAM */
288 uint64_t ullMemSizeVBox;
289 /* If there is a <vbox:Machine>, we always prefer the setting from there. */
290 if ( vsysThis.pelmVBoxMachine
291 && pNewDesc->m->pConfig->hardwareMachine.ulMemorySizeMB)
292 ullMemSizeVBox = pNewDesc->m->pConfig->hardwareMachine.ulMemorySizeMB;
293 else
294 ullMemSizeVBox = vsysThis.ullMemorySize / _1M;
295 /* Check for the constraints */
296 if ( ullMemSizeVBox != 0
297 && ( ullMemSizeVBox < MM_RAM_MIN_IN_MB
298 || ullMemSizeVBox > MM_RAM_MAX_IN_MB
299 )
300 )
301 {
302 i_addWarning(tr("The virtual system \"%s\" claims support for %llu MB RAM size, but VirtualBox has "
303 "support for min %u & max %u MB RAM size only."),
304 vsysThis.strName.c_str(), ullMemSizeVBox, MM_RAM_MIN_IN_MB, MM_RAM_MAX_IN_MB);
305 ullMemSizeVBox = RT_MIN(RT_MAX(ullMemSizeVBox, MM_RAM_MIN_IN_MB), MM_RAM_MAX_IN_MB);
306 }
307 if (vsysThis.ullMemorySize == 0)
308 {
309 /* If the RAM of the OVF is zero, use our predefined values */
310 ULONG memSizeVBox2;
311 rc = pGuestOSType->COMGETTER(RecommendedRAM)(&memSizeVBox2);
312 if (FAILED(rc)) throw rc;
313 /* VBox stores that in MByte */
314 ullMemSizeVBox = (uint64_t)memSizeVBox2;
315 }
316 pNewDesc->i_addEntry(VirtualSystemDescriptionType_Memory,
317 "",
318 Utf8StrFmt("%RU64", (uint64_t)vsysThis.ullMemorySize),
319 Utf8StrFmt("%RU64", (uint64_t)ullMemSizeVBox));
320
321 /* Audio */
322 Utf8Str strSoundCard;
323 Utf8Str strSoundCardOrig;
324 /* If there is a <vbox:Machine>, we always prefer the setting from there. */
325 if ( vsysThis.pelmVBoxMachine
326 && pNewDesc->m->pConfig->hardwareMachine.audioAdapter.fEnabled)
327 {
328 strSoundCard = Utf8StrFmt("%RU32",
329 (uint32_t)pNewDesc->m->pConfig->hardwareMachine.audioAdapter.controllerType);
330 }
331 else if (vsysThis.strSoundCardType.isNotEmpty())
332 {
333 /* Set the AC97 always for the simple OVF case.
334 * @todo: figure out the hardware which could be possible */
335 strSoundCard = Utf8StrFmt("%RU32", (uint32_t)AudioControllerType_AC97);
336 strSoundCardOrig = vsysThis.strSoundCardType;
337 }
338 if (strSoundCard.isNotEmpty())
339 pNewDesc->i_addEntry(VirtualSystemDescriptionType_SoundCard,
340 "",
341 strSoundCardOrig,
342 strSoundCard);
343
344#ifdef VBOX_WITH_USB
345 /* USB Controller */
346 /* If there is a <vbox:Machine>, we always prefer the setting from there. */
347 if ( ( vsysThis.pelmVBoxMachine
348 && pNewDesc->m->pConfig->hardwareMachine.usbSettings.llUSBControllers.size() > 0)
349 || vsysThis.fHasUsbController)
350 pNewDesc->i_addEntry(VirtualSystemDescriptionType_USBController, "", "", "");
351#endif /* VBOX_WITH_USB */
352
353 /* Network Controller */
354 /* If there is a <vbox:Machine>, we always prefer the setting from there. */
355 if (vsysThis.pelmVBoxMachine)
356 {
357 uint32_t maxNetworkAdapters = Global::getMaxNetworkAdapters(pNewDesc->m->pConfig->hardwareMachine.chipsetType);
358
359 const settings::NetworkAdaptersList &llNetworkAdapters = pNewDesc->m->pConfig->hardwareMachine.llNetworkAdapters;
360 /* Check for the constrains */
361 if (llNetworkAdapters.size() > maxNetworkAdapters)
362 i_addWarning(tr("The virtual system \"%s\" claims support for %zu network adapters, but VirtualBox "
363 "has support for max %u network adapter only."),
364 vsysThis.strName.c_str(), llNetworkAdapters.size(), maxNetworkAdapters);
365 /* Iterate through all network adapters. */
366 settings::NetworkAdaptersList::const_iterator it1;
367 size_t a = 0;
368 for (it1 = llNetworkAdapters.begin();
369 it1 != llNetworkAdapters.end() && a < maxNetworkAdapters;
370 ++it1, ++a)
371 {
372 if (it1->fEnabled)
373 {
374 Utf8Str strMode = convertNetworkAttachmentTypeToString(it1->mode);
375 pNewDesc->i_addEntry(VirtualSystemDescriptionType_NetworkAdapter,
376 "", // ref
377 strMode, // orig
378 Utf8StrFmt("%RU32", (uint32_t)it1->type), // conf
379 0,
380 Utf8StrFmt("slot=%RU32;type=%s", it1->ulSlot, strMode.c_str())); // extra conf
381 }
382 }
383 }
384 /* else we use the ovf configuration. */
385 else if (vsysThis.llEthernetAdapters.size() > 0)
386 {
387 size_t cEthernetAdapters = vsysThis.llEthernetAdapters.size();
388 uint32_t maxNetworkAdapters = Global::getMaxNetworkAdapters(ChipsetType_PIIX3);
389
390 /* Check for the constrains */
391 if (cEthernetAdapters > maxNetworkAdapters)
392 i_addWarning(tr("The virtual system \"%s\" claims support for %zu network adapters, but VirtualBox "
393 "has support for max %u network adapter only."),
394 vsysThis.strName.c_str(), cEthernetAdapters, maxNetworkAdapters);
395
396 /* Get the default network adapter type for the selected guest OS */
397 NetworkAdapterType_T defaultAdapterVBox = NetworkAdapterType_Am79C970A;
398 rc = pGuestOSType->COMGETTER(AdapterType)(&defaultAdapterVBox);
399 if (FAILED(rc)) throw rc;
400
401 ovf::EthernetAdaptersList::const_iterator itEA;
402 /* Iterate through all abstract networks. Ignore network cards
403 * which exceed the limit of VirtualBox. */
404 size_t a = 0;
405 for (itEA = vsysThis.llEthernetAdapters.begin();
406 itEA != vsysThis.llEthernetAdapters.end() && a < maxNetworkAdapters;
407 ++itEA, ++a)
408 {
409 const ovf::EthernetAdapter &ea = *itEA; // logical network to connect to
410 Utf8Str strNetwork = ea.strNetworkName;
411 // make sure it's one of these two
412 if ( (strNetwork.compare("Null", Utf8Str::CaseInsensitive))
413 && (strNetwork.compare("NAT", Utf8Str::CaseInsensitive))
414 && (strNetwork.compare("Bridged", Utf8Str::CaseInsensitive))
415 && (strNetwork.compare("Internal", Utf8Str::CaseInsensitive))
416 && (strNetwork.compare("HostOnly", Utf8Str::CaseInsensitive))
417 && (strNetwork.compare("Generic", Utf8Str::CaseInsensitive))
418 )
419 strNetwork = "Bridged"; // VMware assumes this is the default apparently
420
421 /* Figure out the hardware type */
422 NetworkAdapterType_T nwAdapterVBox = defaultAdapterVBox;
423 if (!ea.strAdapterType.compare("PCNet32", Utf8Str::CaseInsensitive))
424 {
425 /* If the default adapter is already one of the two
426 * PCNet adapters use the default one. If not use the
427 * Am79C970A as fallback. */
428 if (!(defaultAdapterVBox == NetworkAdapterType_Am79C970A ||
429 defaultAdapterVBox == NetworkAdapterType_Am79C973))
430 nwAdapterVBox = NetworkAdapterType_Am79C970A;
431 }
432#ifdef VBOX_WITH_E1000
433 /* VMWare accidentally write this with VirtualCenter 3.5,
434 so make sure in this case always to use the VMWare one */
435 else if (!ea.strAdapterType.compare("E10000", Utf8Str::CaseInsensitive))
436 nwAdapterVBox = NetworkAdapterType_I82545EM;
437 else if (!ea.strAdapterType.compare("E1000", Utf8Str::CaseInsensitive))
438 {
439 /* Check if this OVF was written by VirtualBox */
440 if (Utf8Str(vsysThis.strVirtualSystemType).contains("virtualbox", Utf8Str::CaseInsensitive))
441 {
442 /* If the default adapter is already one of the three
443 * E1000 adapters use the default one. If not use the
444 * I82545EM as fallback. */
445 if (!(defaultAdapterVBox == NetworkAdapterType_I82540EM ||
446 defaultAdapterVBox == NetworkAdapterType_I82543GC ||
447 defaultAdapterVBox == NetworkAdapterType_I82545EM))
448 nwAdapterVBox = NetworkAdapterType_I82540EM;
449 }
450 else
451 /* Always use this one since it's what VMware uses */
452 nwAdapterVBox = NetworkAdapterType_I82545EM;
453 }
454#endif /* VBOX_WITH_E1000 */
455
456 pNewDesc->i_addEntry(VirtualSystemDescriptionType_NetworkAdapter,
457 "", // ref
458 ea.strNetworkName, // orig
459 Utf8StrFmt("%RU32", (uint32_t)nwAdapterVBox), // conf
460 0,
461 Utf8StrFmt("type=%s", strNetwork.c_str())); // extra conf
462 }
463 }
464
465 /* If there is a <vbox:Machine>, we always prefer the setting from there. */
466 bool fFloppy = false;
467 bool fDVD = false;
468 if (vsysThis.pelmVBoxMachine)
469 {
470 settings::StorageControllersList &llControllers = pNewDesc->m->pConfig->storageMachine.llStorageControllers;
471 settings::StorageControllersList::iterator it3;
472 for (it3 = llControllers.begin();
473 it3 != llControllers.end();
474 ++it3)
475 {
476 settings::AttachedDevicesList &llAttachments = it3->llAttachedDevices;
477 settings::AttachedDevicesList::iterator it4;
478 for (it4 = llAttachments.begin();
479 it4 != llAttachments.end();
480 ++it4)
481 {
482 fDVD |= it4->deviceType == DeviceType_DVD;
483 fFloppy |= it4->deviceType == DeviceType_Floppy;
484 if (fFloppy && fDVD)
485 break;
486 }
487 if (fFloppy && fDVD)
488 break;
489 }
490 }
491 else
492 {
493 fFloppy = vsysThis.fHasFloppyDrive;
494 fDVD = vsysThis.fHasCdromDrive;
495 }
496 /* Floppy Drive */
497 if (fFloppy)
498 pNewDesc->i_addEntry(VirtualSystemDescriptionType_Floppy, "", "", "");
499 /* CD Drive */
500 if (fDVD)
501 pNewDesc->i_addEntry(VirtualSystemDescriptionType_CDROM, "", "", "");
502
503 /* Hard disk Controller */
504 uint16_t cIDEused = 0;
505 uint16_t cSATAused = 0; NOREF(cSATAused);
506 uint16_t cSCSIused = 0; NOREF(cSCSIused);
507 ovf::ControllersMap::const_iterator hdcIt;
508 /* Iterate through all hard disk controllers */
509 for (hdcIt = vsysThis.mapControllers.begin();
510 hdcIt != vsysThis.mapControllers.end();
511 ++hdcIt)
512 {
513 const ovf::HardDiskController &hdc = hdcIt->second;
514 Utf8Str strControllerID = Utf8StrFmt("%RI32", (uint32_t)hdc.idController);
515
516 switch (hdc.system)
517 {
518 case ovf::HardDiskController::IDE:
519 /* Check for the constrains */
520 if (cIDEused < 4)
521 {
522 // @todo: figure out the IDE types
523 /* Use PIIX4 as default */
524 Utf8Str strType = "PIIX4";
525 if (!hdc.strControllerType.compare("PIIX3", Utf8Str::CaseInsensitive))
526 strType = "PIIX3";
527 else if (!hdc.strControllerType.compare("ICH6", Utf8Str::CaseInsensitive))
528 strType = "ICH6";
529 pNewDesc->i_addEntry(VirtualSystemDescriptionType_HardDiskControllerIDE,
530 strControllerID, // strRef
531 hdc.strControllerType, // aOvfValue
532 strType); // aVBoxValue
533 }
534 else
535 /* Warn only once */
536 if (cIDEused == 2)
537 i_addWarning(tr("The virtual \"%s\" system requests support for more than two "
538 "IDE controller channels, but VirtualBox supports only two."),
539 vsysThis.strName.c_str());
540
541 ++cIDEused;
542 break;
543
544 case ovf::HardDiskController::SATA:
545 /* Check for the constrains */
546 if (cSATAused < 1)
547 {
548 // @todo: figure out the SATA types
549 /* We only support a plain AHCI controller, so use them always */
550 pNewDesc->i_addEntry(VirtualSystemDescriptionType_HardDiskControllerSATA,
551 strControllerID,
552 hdc.strControllerType,
553 "AHCI");
554 }
555 else
556 {
557 /* Warn only once */
558 if (cSATAused == 1)
559 i_addWarning(tr("The virtual system \"%s\" requests support for more than one "
560 "SATA controller, but VirtualBox has support for only one"),
561 vsysThis.strName.c_str());
562
563 }
564 ++cSATAused;
565 break;
566
567 case ovf::HardDiskController::SCSI:
568 /* Check for the constrains */
569 if (cSCSIused < 1)
570 {
571 VirtualSystemDescriptionType_T vsdet = VirtualSystemDescriptionType_HardDiskControllerSCSI;
572 Utf8Str hdcController = "LsiLogic";
573 if (!hdc.strControllerType.compare("lsilogicsas", Utf8Str::CaseInsensitive))
574 {
575 // OVF considers SAS a variant of SCSI but VirtualBox considers it a class of its own
576 vsdet = VirtualSystemDescriptionType_HardDiskControllerSAS;
577 hdcController = "LsiLogicSas";
578 }
579 else if (!hdc.strControllerType.compare("BusLogic", Utf8Str::CaseInsensitive))
580 hdcController = "BusLogic";
581 pNewDesc->i_addEntry(vsdet,
582 strControllerID,
583 hdc.strControllerType,
584 hdcController);
585 }
586 else
587 i_addWarning(tr("The virtual system \"%s\" requests support for an additional "
588 "SCSI controller of type \"%s\" with ID %s, but VirtualBox presently "
589 "supports only one SCSI controller."),
590 vsysThis.strName.c_str(),
591 hdc.strControllerType.c_str(),
592 strControllerID.c_str());
593 ++cSCSIused;
594 break;
595 }
596 }
597
598 /* Hard disks */
599 if (vsysThis.mapVirtualDisks.size() > 0)
600 {
601 ovf::VirtualDisksMap::const_iterator itVD;
602 /* Iterate through all hard disks ()*/
603 for (itVD = vsysThis.mapVirtualDisks.begin();
604 itVD != vsysThis.mapVirtualDisks.end();
605 ++itVD)
606 {
607 const ovf::VirtualDisk &hd = itVD->second;
608 /* Get the associated disk image */
609 ovf::DiskImage di;
610 std::map<RTCString, ovf::DiskImage>::iterator foundDisk;
611
612 foundDisk = m->pReader->m_mapDisks.find(hd.strDiskId);
613 if (foundDisk == m->pReader->m_mapDisks.end())
614 continue;
615 else
616 {
617 di = foundDisk->second;
618 }
619
620 /*
621 * Figure out from URI which format the image of disk has.
622 * URI must have inside section <Disk> .
623 * But there aren't strong requirements about correspondence one URI for one disk virtual format.
624 * So possibly, we aren't able to recognize some URIs.
625 */
626
627 ComObjPtr<MediumFormat> mediumFormat;
628 rc = i_findMediumFormatFromDiskImage(di, mediumFormat);
629 if (FAILED(rc))
630 throw rc;
631
632 Bstr bstrFormatName;
633 rc = mediumFormat->COMGETTER(Name)(bstrFormatName.asOutParam());
634 if (FAILED(rc))
635 throw rc;
636 Utf8Str vdf = Utf8Str(bstrFormatName);
637
638 // @todo:
639 // - figure out all possible vmdk formats we also support
640 // - figure out if there is a url specifier for vhd already
641 // - we need a url specifier for the vdi format
642
643 if (vdf.compare("VMDK", Utf8Str::CaseInsensitive) == 0)
644 {
645 /* If the href is empty use the VM name as filename */
646 Utf8Str strFilename = di.strHref;
647 if (!strFilename.length())
648 strFilename = Utf8StrFmt("%s.vmdk", hd.strDiskId.c_str());
649
650 Utf8Str strTargetPath = Utf8Str(strMachineFolder);
651 strTargetPath.append(RTPATH_DELIMITER).append(di.strHref);
652 /*
653 * Remove last extension from the file name if the file is compressed
654 */
655 if (di.strCompression.compare("gzip", Utf8Str::CaseInsensitive)==0)
656 {
657 strTargetPath.stripSuffix();
658 }
659
660 i_searchUniqueDiskImageFilePath(strTargetPath);
661
662 /* find the description for the hard disk controller
663 * that has the same ID as hd.idController */
664 const VirtualSystemDescriptionEntry *pController;
665 if (!(pController = pNewDesc->i_findControllerFromID(hd.idController)))
666 throw setError(E_FAIL,
667 tr("Cannot find hard disk controller with OVF instance ID %RI32 "
668 "to which disk \"%s\" should be attached"),
669 hd.idController,
670 di.strHref.c_str());
671
672 /* controller to attach to, and the bus within that controller */
673 Utf8StrFmt strExtraConfig("controller=%RI16;channel=%RI16",
674 pController->ulIndex,
675 hd.ulAddressOnParent);
676 pNewDesc->i_addEntry(VirtualSystemDescriptionType_HardDiskImage,
677 hd.strDiskId,
678 di.strHref,
679 strTargetPath,
680 di.ulSuggestedSizeMB,
681 strExtraConfig);
682 }
683 else if (vdf.compare("RAW", Utf8Str::CaseInsensitive) == 0)
684 {
685 /* If the href is empty use the VM name as filename */
686 Utf8Str strFilename = di.strHref;
687 if (!strFilename.length())
688 strFilename = Utf8StrFmt("%s.iso", hd.strDiskId.c_str());
689
690 Utf8Str strTargetPath = Utf8Str(strMachineFolder)
691 .append(RTPATH_DELIMITER)
692 .append(di.strHref);
693 /*
694 * Remove last extension from the file name if the file is compressed
695 */
696 if (di.strCompression.compare("gzip", Utf8Str::CaseInsensitive)==0)
697 {
698 strTargetPath.stripSuffix();
699 }
700
701 i_searchUniqueDiskImageFilePath(strTargetPath);
702
703 /* find the description for the hard disk controller
704 * that has the same ID as hd.idController */
705 const VirtualSystemDescriptionEntry *pController;
706 if (!(pController = pNewDesc->i_findControllerFromID(hd.idController)))
707 throw setError(E_FAIL,
708 tr("Cannot find disk controller with OVF instance ID %RI32 "
709 "to which disk \"%s\" should be attached"),
710 hd.idController,
711 di.strHref.c_str());
712
713 /* controller to attach to, and the bus within that controller */
714 Utf8StrFmt strExtraConfig("controller=%RI16;channel=%RI16",
715 pController->ulIndex,
716 hd.ulAddressOnParent);
717 pNewDesc->i_addEntry(VirtualSystemDescriptionType_HardDiskImage,
718 hd.strDiskId,
719 di.strHref,
720 strTargetPath,
721 di.ulSuggestedSizeMB,
722 strExtraConfig);
723 }
724 else
725 throw setError(VBOX_E_FILE_ERROR,
726 tr("Unsupported format for virtual disk image %s in OVF: \"%s\""),
727 di.strHref.c_str(),
728 di.strFormat.c_str());
729 }
730 }
731
732 m->virtualSystemDescriptions.push_back(pNewDesc);
733 }
734 }
735 catch (HRESULT aRC)
736 {
737 /* On error we clear the list & return */
738 m->virtualSystemDescriptions.clear();
739 rc = aRC;
740 }
741
742 // reset the appliance state
743 alock.acquire();
744 m->state = Data::ApplianceIdle;
745
746 return rc;
747}
748
749/**
750 * Public method implementation. This creates one or more new machines according to the
751 * VirtualSystemScription instances created by Appliance::Interpret().
752 * Thread implementation is in Appliance::i_importImpl().
753 * @param aProgress
754 * @return
755 */
756HRESULT Appliance::importMachines(const std::vector<ImportOptions_T> &aOptions,
757 ComPtr<IProgress> &aProgress)
758{
759 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
760
761 if (aOptions.size())
762 {
763 m->optListImport.setCapacity(aOptions.size());
764 for (size_t i = 0; i < aOptions.size(); ++i)
765 {
766 m->optListImport.insert(i, aOptions[i]);
767 }
768 }
769
770 AssertReturn(!( m->optListImport.contains(ImportOptions_KeepAllMACs)
771 && m->optListImport.contains(ImportOptions_KeepNATMACs) )
772 , E_INVALIDARG);
773
774 // do not allow entering this method if the appliance is busy reading or writing
775 if (!i_isApplianceIdle())
776 return E_ACCESSDENIED;
777
778 if (!m->pReader)
779 return setError(E_FAIL,
780 tr("Cannot import machines without reading it first (call read() before i_importMachines())"));
781
782 ComObjPtr<Progress> progress;
783 HRESULT rc = S_OK;
784 try
785 {
786 rc = i_importImpl(m->locInfo, progress);
787 }
788 catch (HRESULT aRC)
789 {
790 rc = aRC;
791 }
792
793 if (SUCCEEDED(rc))
794 /* Return progress to the caller */
795 progress.queryInterfaceTo(aProgress.asOutParam());
796
797 return rc;
798}
799
800////////////////////////////////////////////////////////////////////////////////
801//
802// Appliance private methods
803//
804////////////////////////////////////////////////////////////////////////////////
805
806/**
807 * Ensures that there is a look-ahead object ready.
808 *
809 * @returns true if there's an object handy, false if end-of-stream.
810 * @throws HRESULT if the next object isn't a regular file. Sets error info
811 * (which is why it's a method on Appliance and not the
812 * ImportStack).
813 */
814bool Appliance::i_importEnsureOvaLookAhead(ImportStack &stack)
815{
816 Assert(stack.hVfsFssOva != NULL);
817 if (stack.hVfsIosOvaLookAhead == NIL_RTVFSIOSTREAM)
818 {
819 RTStrFree(stack.pszOvaLookAheadName);
820 stack.pszOvaLookAheadName = NULL;
821
822 RTVFSOBJTYPE enmType;
823 RTVFSOBJ hVfsObj;
824 int vrc = RTVfsFsStrmNext(stack.hVfsFssOva, &stack.pszOvaLookAheadName, &enmType, &hVfsObj);
825 if (RT_SUCCESS(vrc))
826 {
827 stack.hVfsIosOvaLookAhead = RTVfsObjToIoStream(hVfsObj);
828 RTVfsObjRelease(hVfsObj);
829 if ( ( enmType != RTVFSOBJTYPE_FILE
830 && enmType != RTVFSOBJTYPE_IO_STREAM)
831 || stack.hVfsIosOvaLookAhead == NIL_RTVFSIOSTREAM)
832 throw setError(VBOX_E_FILE_ERROR,
833 tr("Malformed OVA. '%s' is not a regular file (%d)."), stack.pszOvaLookAheadName, enmType);
834 }
835 else if (vrc == VERR_EOF)
836 return false;
837 else
838 throw setErrorVrc(vrc, tr("RTVfsFsStrmNext failed (%Rrc)"), vrc);
839 }
840 return true;
841}
842
843HRESULT Appliance::i_preCheckImageAvailability(ImportStack &stack)
844{
845 if (i_importEnsureOvaLookAhead(stack))
846 return S_OK;
847 throw setError(VBOX_E_FILE_ERROR, tr("Unexpected end of OVA package"));
848 /** @todo r=bird: dunno why this bother returning a value and the caller
849 * having a special 'continue' case for it. It always threw all non-OK
850 * status codes. It's possibly to handle out of order stuff, so that
851 * needs adding to the testcase! */
852}
853
854/**
855 * Setup automatic I/O stream digest calculation, adding it to hOurManifest.
856 *
857 * @returns Passthru I/O stream, of @a hVfsIos if no digest calc needed.
858 * @param hVfsIos The stream to wrap. Always consumed.
859 * @param pszManifestEntry The manifest entry.
860 * @throws Nothing.
861 */
862RTVFSIOSTREAM Appliance::i_importSetupDigestCalculationForGivenIoStream(RTVFSIOSTREAM hVfsIos, const char *pszManifestEntry)
863{
864 int vrc;
865 Assert(!RTManifestPtIosIsInstanceOf(hVfsIos));
866
867 if (m->fDigestTypes == 0)
868 return hVfsIos;
869
870 /* Create the manifest if necessary. */
871 if (m->hOurManifest == NIL_RTMANIFEST)
872 {
873 vrc = RTManifestCreate(0 /*fFlags*/, &m->hOurManifest);
874 AssertRCReturnStmt(vrc, RTVfsIoStrmRelease(hVfsIos), NIL_RTVFSIOSTREAM);
875 }
876
877 /* Setup the stream. */
878 RTVFSIOSTREAM hVfsIosPt;
879 vrc = RTManifestEntryAddPassthruIoStream(m->hOurManifest, hVfsIos, pszManifestEntry, m->fDigestTypes,
880 true /*fReadOrWrite*/, &hVfsIosPt);
881
882 RTVfsIoStrmRelease(hVfsIos); /* always consumed! */
883 if (RT_SUCCESS(vrc))
884 return hVfsIosPt;
885
886 setErrorVrc(vrc, "RTManifestEntryAddPassthruIoStream failed with rc=%Rrc", vrc);
887 return NIL_RTVFSIOSTREAM;
888}
889
890/**
891 * Opens a source file (for reading obviously).
892 *
893 * @param rstrSrcPath The source file to open.
894 * @param pszManifestEntry The manifest entry of the source file. This is
895 * used when constructing our manifest using a pass
896 * thru.
897 * @returns I/O stream handle to the source file.
898 * @throws HRESULT error status, error info set.
899 */
900RTVFSIOSTREAM Appliance::i_importOpenSourceFile(ImportStack &stack, Utf8Str const &rstrSrcPath, const char *pszManifestEntry)
901{
902 /*
903 * Open the source file. Special considerations for OVAs.
904 */
905 RTVFSIOSTREAM hVfsIosSrc;
906 if (stack.hVfsFssOva != NIL_RTVFSFSSTREAM)
907 {
908 for (uint32_t i = 0;; i++)
909 {
910 if (!i_importEnsureOvaLookAhead(stack))
911 throw setErrorBoth(VBOX_E_FILE_ERROR, VERR_EOF,
912 tr("Unexpected end of OVA / internal error - missing '%s' (skipped %u)"),
913 rstrSrcPath.c_str(), i);
914 if (RTStrICmp(stack.pszOvaLookAheadName, rstrSrcPath.c_str()) == 0)
915 break;
916
917 /* release the current object, loop to get the next. */
918 RTVfsIoStrmRelease(stack.claimOvaLookAHead());
919 }
920 hVfsIosSrc = stack.claimOvaLookAHead();
921 }
922 else
923 {
924 int vrc = RTVfsIoStrmOpenNormal(rstrSrcPath.c_str(), RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_NONE, &hVfsIosSrc);
925 if (RT_FAILURE(vrc))
926 throw setErrorVrc(vrc, tr("Error opening '%s' for reading (%Rrc)"), rstrSrcPath.c_str(), vrc);
927 }
928
929 /*
930 * Digest calculation filtering.
931 */
932 hVfsIosSrc = i_importSetupDigestCalculationForGivenIoStream(hVfsIosSrc, pszManifestEntry);
933 if (hVfsIosSrc == NIL_RTVFSIOSTREAM)
934 throw E_FAIL;
935
936 return hVfsIosSrc;
937}
938
939/**
940 * Creates the destination file and fills it with bytes from the source stream.
941 *
942 * This assumes that we digest the source when fDigestTypes is non-zero, and
943 * thus calls RTManifestPtIosAddEntryNow when done.
944 *
945 * @param rstrDstPath The path to the destination file. Missing path
946 * components will be created.
947 * @param hVfsIosSrc The source I/O stream.
948 * @param rstrSrcLogNm The name of the source for logging and error
949 * messages.
950 * @returns COM status code.
951 * @throws Nothing (as the caller has VFS handles to release).
952 */
953HRESULT Appliance::i_importCreateAndWriteDestinationFile(Utf8Str const &rstrDstPath, RTVFSIOSTREAM hVfsIosSrc,
954 Utf8Str const &rstrSrcLogNm)
955{
956 int vrc;
957
958 /*
959 * Create the output file, including necessary paths.
960 * Any existing file will be overwritten.
961 */
962 HRESULT hrc = VirtualBox::i_ensureFilePathExists(rstrDstPath, true /*fCreate*/);
963 if (SUCCEEDED(hrc))
964 {
965 RTVFSIOSTREAM hVfsIosDst;
966 vrc = RTVfsIoStrmOpenNormal(rstrDstPath.c_str(),
967 RTFILE_O_CREATE_REPLACE | RTFILE_O_WRITE | RTFILE_O_DENY_ALL,
968 &hVfsIosDst);
969 if (RT_SUCCESS(vrc))
970 {
971 /*
972 * Pump the bytes thru. If we fail, delete the output file.
973 */
974 vrc = RTVfsUtilPumpIoStreams(hVfsIosSrc, hVfsIosDst, 0);
975 if (RT_SUCCESS(vrc))
976 hrc = S_OK;
977 else
978 hrc = setErrorVrc(vrc, tr("Error occured decompressing '%s' to '%s' (%Rrc)"),
979 rstrSrcLogNm.c_str(), rstrDstPath.c_str(), vrc);
980 uint32_t cRefs = RTVfsIoStrmRelease(hVfsIosDst);
981 AssertMsg(cRefs == 0, ("cRefs=%u\n", cRefs)); NOREF(cRefs);
982 if (RT_FAILURE(vrc))
983 RTFileDelete(rstrDstPath.c_str());
984 }
985 else
986 hrc = setErrorVrc(vrc, tr("Error opening destionation image '%s' for writing (%Rrc)"), rstrDstPath.c_str(), vrc);
987 }
988 return hrc;
989}
990
991
992/**
993 *
994 * @param pszManifestEntry The manifest entry of the source file. This is
995 * used when constructing our manifest using a pass
996 * thru.
997 * @throws HRESULT error status, error info set.
998 */
999void Appliance::i_importCopyFile(ImportStack &stack, Utf8Str const &rstrSrcPath, Utf8Str const &rstrDstPath,
1000 const char *pszManifestEntry)
1001{
1002 /*
1003 * Open the file (throws error) and add a read ahead thread so we can do
1004 * concurrent reads (+digest) and writes.
1005 */
1006 RTVFSIOSTREAM hVfsIosSrc = i_importOpenSourceFile(stack, rstrSrcPath, pszManifestEntry);
1007 RTVFSIOSTREAM hVfsIosReadAhead;
1008 int vrc = RTVfsCreateReadAheadForIoStream(hVfsIosSrc, 0 /*fFlags*/, 0 /*cBuffers=default*/, 0 /*cbBuffers=default*/,
1009 &hVfsIosReadAhead);
1010 if (RT_FAILURE(vrc))
1011 {
1012 RTVfsIoStrmRelease(hVfsIosSrc);
1013 throw setErrorVrc(vrc, tr("Error initializing read ahead thread for '%s' (%Rrc)"), rstrSrcPath.c_str(), vrc);
1014 }
1015
1016 /*
1017 * Write the destination file (nothrow).
1018 */
1019 HRESULT hrc = i_importCreateAndWriteDestinationFile(rstrDstPath, hVfsIosReadAhead, rstrSrcPath);
1020 RTVfsIoStrmRelease(hVfsIosReadAhead);
1021
1022 /*
1023 * Before releasing the source stream, make sure we've successfully added
1024 * the digest to our manifest.
1025 */
1026 if (SUCCEEDED(hrc) && m->fDigestTypes)
1027 {
1028 vrc = RTManifestPtIosAddEntryNow(hVfsIosSrc);
1029 if (RT_FAILURE(vrc))
1030 hrc = setErrorVrc(vrc, tr("RTManifestPtIosAddEntryNow failed with %Rrc"), vrc);
1031 }
1032
1033 uint32_t cRefs = RTVfsIoStrmRelease(hVfsIosSrc);
1034 AssertMsg(cRefs == 0, ("cRefs=%u\n", cRefs)); NOREF(cRefs);
1035 if (SUCCEEDED(hrc))
1036 return;
1037 throw hrc;
1038}
1039
1040/**
1041 *
1042 * @param pszManifestEntry The manifest entry of the source file. This is
1043 * used when constructing our manifest using a pass
1044 * thru.
1045 * @throws HRESULT error status, error info set.
1046 */
1047void Appliance::i_importDecompressFile(ImportStack &stack, Utf8Str const &rstrSrcPath, Utf8Str const &rstrDstPath,
1048 const char *pszManifestEntry)
1049{
1050 RTVFSIOSTREAM hVfsIosSrcCompressed = i_importOpenSourceFile(stack, rstrSrcPath, pszManifestEntry);
1051
1052 /*
1053 * Add a read ahead thread here. This means reading and digest calculation
1054 * is done on one thread, while unpacking and writing is one on this thread.
1055 */
1056 RTVFSIOSTREAM hVfsIosReadAhead;
1057 int vrc = RTVfsCreateReadAheadForIoStream(hVfsIosSrcCompressed, 0 /*fFlags*/, 0 /*cBuffers=default*/,
1058 0 /*cbBuffers=default*/, &hVfsIosReadAhead);
1059 if (RT_FAILURE(vrc))
1060 {
1061 RTVfsIoStrmRelease(hVfsIosSrcCompressed);
1062 throw setErrorVrc(vrc, tr("Error initializing read ahead thread for '%s' (%Rrc)"), rstrSrcPath.c_str(), vrc);
1063 }
1064
1065 /*
1066 * Add decompression step.
1067 */
1068 RTVFSIOSTREAM hVfsIosSrc;
1069 vrc = RTZipGzipDecompressIoStream(hVfsIosReadAhead, 0, &hVfsIosSrc);
1070 RTVfsIoStrmRelease(hVfsIosReadAhead);
1071 if (RT_FAILURE(vrc))
1072 {
1073 RTVfsIoStrmRelease(hVfsIosSrcCompressed);
1074 throw setErrorVrc(vrc, tr("Error initializing gzip decompression for '%s' (%Rrc)"), rstrSrcPath.c_str(), vrc);
1075 }
1076
1077 /*
1078 * Write the stream to the destination file (nothrow).
1079 */
1080 HRESULT hrc = i_importCreateAndWriteDestinationFile(rstrDstPath, hVfsIosSrc, rstrSrcPath);
1081
1082 /*
1083 * Before releasing the source stream, make sure we've successfully added
1084 * the digest to our manifest.
1085 */
1086 if (SUCCEEDED(hrc) && m->fDigestTypes)
1087 {
1088 vrc = RTManifestPtIosAddEntryNow(hVfsIosSrcCompressed);
1089 if (RT_FAILURE(vrc))
1090 hrc = setErrorVrc(vrc, tr("RTManifestPtIosAddEntryNow failed with %Rrc"), vrc);
1091 }
1092
1093 uint32_t cRefs = RTVfsIoStrmRelease(hVfsIosSrc);
1094 AssertMsg(cRefs == 0, ("cRefs=%u\n", cRefs)); NOREF(cRefs);
1095
1096 cRefs = RTVfsIoStrmRelease(hVfsIosSrcCompressed);
1097 AssertMsg(cRefs == 0, ("cRefs=%u\n", cRefs)); NOREF(cRefs);
1098
1099 if (SUCCEEDED(hrc))
1100 return;
1101 throw hrc;
1102}
1103
1104/*******************************************************************************
1105 * Read stuff
1106 ******************************************************************************/
1107
1108/**
1109 * Implementation for reading an OVF (via task).
1110 *
1111 * This starts a new thread which will call
1112 * Appliance::taskThreadImportOrExport() which will then call readFS(). This
1113 * will then open the OVF with ovfreader.cpp.
1114 *
1115 * This is in a separate private method because it is used from two locations:
1116 *
1117 * 1) from the public Appliance::Read().
1118 *
1119 * 2) in a second worker thread; in that case, Appliance::ImportMachines() called Appliance::i_importImpl(), which
1120 * called Appliance::readFSOVA(), which called Appliance::i_importImpl(), which then called this again.
1121 *
1122 * @param aLocInfo The OVF location.
1123 * @param aProgress Where to return the progress object.
1124 * @throws COM error codes will be thrown.
1125 */
1126void Appliance::i_readImpl(const LocationInfo &aLocInfo, ComObjPtr<Progress> &aProgress)
1127{
1128 BstrFmt bstrDesc = BstrFmt(tr("Reading appliance '%s'"),
1129 aLocInfo.strPath.c_str());
1130 HRESULT rc;
1131 /* Create the progress object */
1132 aProgress.createObject();
1133 if (aLocInfo.storageType == VFSType_File)
1134 /* 1 operation only */
1135 rc = aProgress->init(mVirtualBox, static_cast<IAppliance*>(this),
1136 bstrDesc.raw(),
1137 TRUE /* aCancelable */);
1138 else
1139 /* 4/5 is downloading, 1/5 is reading */
1140 rc = aProgress->init(mVirtualBox, static_cast<IAppliance*>(this),
1141 bstrDesc.raw(),
1142 TRUE /* aCancelable */,
1143 2, // ULONG cOperations,
1144 5, // ULONG ulTotalOperationsWeight,
1145 BstrFmt(tr("Download appliance '%s'"),
1146 aLocInfo.strPath.c_str()).raw(), // CBSTR bstrFirstOperationDescription,
1147 4); // ULONG ulFirstOperationWeight,
1148 if (FAILED(rc)) throw rc;
1149
1150 /* Initialize our worker task */
1151 TaskOVF *task = NULL;
1152 try
1153 {
1154 task = new TaskOVF(this, TaskOVF::Read, aLocInfo, aProgress);
1155 }
1156 catch (...)
1157 {
1158 throw setError(VBOX_E_OBJECT_NOT_FOUND,
1159 tr("Could not create TaskOVF object for reading the OVF from disk"));
1160 }
1161
1162 rc = task->createThread();
1163 if (FAILED(rc)) throw rc;
1164}
1165
1166/**
1167 * Actual worker code for reading an OVF from disk. This is called from Appliance::taskThreadImportOrExport()
1168 * and therefore runs on the OVF read worker thread. This opens the OVF with ovfreader.cpp.
1169 *
1170 * This runs in one context:
1171 *
1172 * 1) in a first worker thread; in that case, Appliance::Read() called Appliance::readImpl();
1173 *
1174 * @param pTask
1175 * @return
1176 */
1177HRESULT Appliance::i_readFS(TaskOVF *pTask)
1178{
1179 LogFlowFuncEnter();
1180 LogFlowFunc(("Appliance %p\n", this));
1181
1182 AutoCaller autoCaller(this);
1183 if (FAILED(autoCaller.rc())) return autoCaller.rc();
1184
1185 AutoWriteLock appLock(this COMMA_LOCKVAL_SRC_POS);
1186
1187 HRESULT rc;
1188 if (pTask->locInfo.strPath.endsWith(".ovf", Utf8Str::CaseInsensitive))
1189 rc = i_readFSOVF(pTask);
1190 else
1191 rc = i_readFSOVA(pTask);
1192
1193 LogFlowFunc(("rc=%Rhrc\n", rc));
1194 LogFlowFuncLeave();
1195
1196 return rc;
1197}
1198
1199HRESULT Appliance::i_readFSOVF(TaskOVF *pTask)
1200{
1201 LogFlowFunc(("'%s'\n", pTask->locInfo.strPath.c_str()));
1202
1203 /*
1204 * Allocate a buffer for filenames and prep it for suffix appending.
1205 */
1206 char *pszNameBuf = (char *)alloca(pTask->locInfo.strPath.length() + 16);
1207 AssertReturn(pszNameBuf, VERR_NO_TMP_MEMORY);
1208 memcpy(pszNameBuf, pTask->locInfo.strPath.c_str(), pTask->locInfo.strPath.length() + 1);
1209 RTPathStripSuffix(pszNameBuf);
1210 size_t const cchBaseName = strlen(pszNameBuf);
1211
1212 /*
1213 * Open the OVF file first since that is what this is all about.
1214 */
1215 RTVFSIOSTREAM hIosOvf;
1216 int vrc = RTVfsIoStrmOpenNormal(pTask->locInfo.strPath.c_str(),
1217 RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_NONE, &hIosOvf);
1218 if (RT_FAILURE(vrc))
1219 return setErrorVrc(vrc, tr("Failed to open OVF file '%s' (%Rrc)"), pTask->locInfo.strPath.c_str(), vrc);
1220
1221 HRESULT hrc = i_readOVFFile(pTask, hIosOvf, RTPathFilename(pTask->locInfo.strPath.c_str())); /* consumes hIosOvf */
1222 if (FAILED(hrc))
1223 return hrc;
1224
1225 /*
1226 * Try open the manifest file (for signature purposes and to determine digest type(s)).
1227 */
1228 RTVFSIOSTREAM hIosMf;
1229 strcpy(&pszNameBuf[cchBaseName], ".mf");
1230 vrc = RTVfsIoStrmOpenNormal(pszNameBuf, RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_NONE, &hIosMf);
1231 if (RT_SUCCESS(vrc))
1232 {
1233 const char * const pszFilenamePart = RTPathFilename(pszNameBuf);
1234 hrc = i_readManifestFile(pTask, hIosMf /*consumed*/, pszFilenamePart);
1235 if (FAILED(hrc))
1236 return hrc;
1237
1238 /*
1239 * Check for the signature file.
1240 */
1241 RTVFSIOSTREAM hIosCert;
1242 strcpy(&pszNameBuf[cchBaseName], ".cert");
1243 vrc = RTVfsIoStrmOpenNormal(pszNameBuf, RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_NONE, &hIosCert);
1244 if (RT_SUCCESS(vrc))
1245 {
1246 hrc = i_readSignatureFile(pTask, hIosCert /*consumed*/, pszFilenamePart);
1247 if (FAILED(hrc))
1248 return hrc;
1249 }
1250 else if (vrc != VERR_FILE_NOT_FOUND && vrc != VERR_PATH_NOT_FOUND)
1251 return setErrorVrc(vrc, tr("Failed to open the signature file '%s' (%Rrc)"), pszNameBuf, vrc);
1252
1253 }
1254 else if (vrc == VERR_FILE_NOT_FOUND || vrc == VERR_PATH_NOT_FOUND)
1255 {
1256 m->fDeterminedDigestTypes = true;
1257 m->fDigestTypes = 0;
1258 }
1259 else
1260 return setErrorVrc(vrc, tr("Failed to open the manifest file '%s' (%Rrc)"), pszNameBuf, vrc);
1261
1262 /*
1263 * Do tail processing (check the signature).
1264 */
1265 hrc = i_readTailProcessing(pTask);
1266
1267 LogFlowFunc(("returns %Rhrc\n", hrc));
1268 return hrc;
1269}
1270
1271HRESULT Appliance::i_readFSOVA(TaskOVF *pTask)
1272{
1273 LogFlowFunc(("'%s'\n", pTask->locInfo.strPath.c_str()));
1274
1275 /*
1276 * Open the tar file as file stream.
1277 */
1278 RTVFSIOSTREAM hVfsIosOva;
1279 int vrc = RTVfsIoStrmOpenNormal(pTask->locInfo.strPath.c_str(),
1280 RTFILE_O_READ | RTFILE_O_DENY_NONE | RTFILE_O_OPEN, &hVfsIosOva);
1281 if (RT_FAILURE(vrc))
1282 return setErrorVrc(vrc, tr("Error opening the OVA file '%s' (%Rrc)"), pTask->locInfo.strPath.c_str(), vrc);
1283
1284 RTVFSFSSTREAM hVfsFssOva;
1285 vrc = RTZipTarFsStreamFromIoStream(hVfsIosOva, 0 /*fFlags*/, &hVfsFssOva);
1286 RTVfsIoStrmRelease(hVfsIosOva);
1287 if (RT_FAILURE(vrc))
1288 return setErrorVrc(vrc, tr("Error reading the OVA file '%s' (%Rrc)"), pTask->locInfo.strPath.c_str(), vrc);
1289
1290 /*
1291 * Since jumping thru an OVA file with seekable disk backing is rather
1292 * efficient, we can process .ovf, .mf and .cert files here without any
1293 * strict ordering restrictions.
1294 *
1295 * (Technically, the .ovf-file comes first, while the manifest and its
1296 * optional signature file either follows immediately or at the very end of
1297 * the OVA. The manifest is optional.)
1298 */
1299 char *pszOvfNameBase = NULL;
1300 size_t cchOvfNameBase = 0;
1301 unsigned cLeftToFind = 3;
1302 HRESULT hrc = S_OK;
1303 do
1304 {
1305 char *pszName = NULL;
1306 RTVFSOBJTYPE enmType;
1307 RTVFSOBJ hVfsObj;
1308 vrc = RTVfsFsStrmNext(hVfsFssOva, &pszName, &enmType, &hVfsObj);
1309 if (RT_FAILURE(vrc))
1310 {
1311 if (vrc != VERR_EOF)
1312 hrc = setErrorVrc(vrc, tr("Error reading OVA '%s' (%Rrc)"), pTask->locInfo.strPath.c_str(), vrc);
1313 break;
1314 }
1315
1316 /* We only care about entries that are files. Get the I/O stream handle for them. */
1317 if ( enmType == RTVFSOBJTYPE_IO_STREAM
1318 || enmType == RTVFSOBJTYPE_FILE)
1319 {
1320 /* Find the suffix and check if this is a possibly interesting file. */
1321 char *pszSuffix = strrchr(pszName, '.');
1322 if ( pszSuffix
1323 && ( RTStrICmp(pszSuffix + 1, "ovf") == 0
1324 || RTStrICmp(pszSuffix + 1, "mf") == 0
1325 || RTStrICmp(pszSuffix + 1, "cert") == 0) )
1326 {
1327 /* Match the OVF base name. */
1328 *pszSuffix = '\0';
1329 if ( pszOvfNameBase == NULL
1330 || RTStrICmp(pszName, pszOvfNameBase) == 0)
1331 {
1332 *pszSuffix = '.';
1333
1334 /* Since we're pretty sure we'll be processing this file, get the I/O stream. */
1335 RTVFSIOSTREAM hVfsIos = RTVfsObjToIoStream(hVfsObj);
1336 Assert(hVfsIos != NIL_RTVFSIOSTREAM);
1337
1338 /* Check for the OVF (should come first). */
1339 if (RTStrICmp(pszSuffix + 1, "ovf") == 0)
1340 {
1341 if (pszOvfNameBase == NULL)
1342 {
1343 hrc = i_readOVFFile(pTask, hVfsIos, pszName);
1344 hVfsIos = NIL_RTVFSIOSTREAM;
1345
1346 /* Set the base name. */
1347 *pszSuffix = '\0';
1348 pszOvfNameBase = pszName;
1349 cchOvfNameBase = strlen(pszName);
1350 pszName = NULL;
1351 cLeftToFind--;
1352 }
1353 else
1354 LogRel(("i_readFSOVA: '%s' contains more than one OVF file ('%s'), picking the first one\n",
1355 pTask->locInfo.strPath.c_str(), pszName));
1356 }
1357 /* Check for manifest. */
1358 else if (RTStrICmp(pszSuffix + 1, "mf") == 0)
1359 {
1360 if (m->hMemFileTheirManifest == NIL_RTVFSFILE)
1361 {
1362 hrc = i_readManifestFile(pTask, hVfsIos, pszName);
1363 hVfsIos = NIL_RTVFSIOSTREAM; /*consumed*/
1364 cLeftToFind--;
1365 }
1366 else
1367 LogRel(("i_readFSOVA: '%s' contains more than one manifest file ('%s'), picking the first one\n",
1368 pTask->locInfo.strPath.c_str(), pszName));
1369 }
1370 /* Check for signature. */
1371 else if (RTStrICmp(pszSuffix + 1, "cert") == 0)
1372 {
1373 if (!m->fSignerCertLoaded)
1374 {
1375 hrc = i_readSignatureFile(pTask, hVfsIos, pszName);
1376 hVfsIos = NIL_RTVFSIOSTREAM; /*consumed*/
1377 cLeftToFind--;
1378 }
1379 else
1380 LogRel(("i_readFSOVA: '%s' contains more than one signature file ('%s'), picking the first one\n",
1381 pTask->locInfo.strPath.c_str(), pszName));
1382 }
1383 else
1384 AssertFailed();
1385 if (hVfsIos != NIL_RTVFSIOSTREAM)
1386 RTVfsIoStrmRelease(hVfsIos);
1387 }
1388 }
1389 }
1390 RTVfsObjRelease(hVfsObj);
1391 RTStrFree(pszName);
1392 } while (cLeftToFind > 0 && SUCCEEDED(hrc));
1393
1394 RTVfsFsStrmRelease(hVfsFssOva);
1395 RTStrFree(pszOvfNameBase);
1396
1397 /*
1398 * Check that we found and OVF file.
1399 */
1400 if (SUCCEEDED(hrc) && !pszOvfNameBase)
1401 hrc = setError(VBOX_E_FILE_ERROR, tr("OVA '%s' does not contain an .ovf-file"), pTask->locInfo.strPath.c_str());
1402 if (SUCCEEDED(hrc))
1403 {
1404 /*
1405 * Do tail processing (check the signature).
1406 */
1407 hrc = i_readTailProcessing(pTask);
1408 }
1409 LogFlowFunc(("returns %Rhrc\n", hrc));
1410 return hrc;
1411}
1412
1413/**
1414 * Reads & parses the OVF file.
1415 *
1416 * @param pTask The read task.
1417 * @param hVfsIosOvf The I/O stream for the OVF. The reference is
1418 * always consumed.
1419 * @param pszManifestEntry The manifest entry name.
1420 * @returns COM status code, error info set.
1421 * @throws Nothing
1422 */
1423HRESULT Appliance::i_readOVFFile(TaskOVF *pTask, RTVFSIOSTREAM hVfsIosOvf, const char *pszManifestEntry)
1424{
1425 LogFlowFunc(("%s[%s]\n", pTask->locInfo.strPath.c_str(), pszManifestEntry));
1426
1427 /*
1428 * Set the OVF manifest entry name (needed for tweaking the manifest
1429 * validation during import).
1430 */
1431 try { m->strOvfManifestEntry = pszManifestEntry; }
1432 catch (...) { return E_OUTOFMEMORY; }
1433
1434 /*
1435 * Set up digest calculation.
1436 */
1437 hVfsIosOvf = i_importSetupDigestCalculationForGivenIoStream(hVfsIosOvf, pszManifestEntry);
1438 if (hVfsIosOvf == NIL_RTVFSIOSTREAM)
1439 return VBOX_E_FILE_ERROR;
1440
1441 /*
1442 * Read the OVF into a memory buffer and parse it.
1443 */
1444 void *pvBufferedOvf;
1445 size_t cbBufferedOvf;
1446 int vrc = RTVfsIoStrmReadAll(hVfsIosOvf, &pvBufferedOvf, &cbBufferedOvf);
1447 uint32_t cRefs = RTVfsIoStrmRelease(hVfsIosOvf); /* consumes stream handle. */
1448 Assert(cRefs == 0);
1449 if (RT_FAILURE(vrc))
1450 return setErrorVrc(vrc, tr("Could not read the OVF file for '%s' (%Rrc)"), pTask->locInfo.strPath.c_str(), vrc);
1451
1452 HRESULT hrc;
1453 try
1454 {
1455 m->pReader = new ovf::OVFReader(pvBufferedOvf, cbBufferedOvf, pTask->locInfo.strPath);
1456 hrc = S_OK;
1457 }
1458 catch (RTCError &rXcpt) // includes all XML exceptions
1459 {
1460 hrc = setError(VBOX_E_FILE_ERROR, rXcpt.what());
1461 }
1462 catch (HRESULT aRC)
1463 {
1464 hrc = aRC;
1465 }
1466 catch (...)
1467 {
1468 hrc = E_FAIL;
1469 }
1470 LogFlowFunc(("OVFReader(%s) -> rc=%Rhrc\n", pTask->locInfo.strPath.c_str(), hrc));
1471
1472 RTVfsIoStrmReadAllFree(pvBufferedOvf, cbBufferedOvf);
1473 if (SUCCEEDED(hrc))
1474 {
1475 /*
1476 * If we see an OVF v2.0 envelope, select only the SHA-256 digest.
1477 */
1478 if ( !m->fDeterminedDigestTypes
1479 && m->pReader->m_envelopeData.getOVFVersion() == ovf::OVFVersion_2_0)
1480 m->fDigestTypes &= ~RTMANIFEST_ATTR_SHA256;
1481 }
1482
1483 return hrc;
1484}
1485
1486/**
1487 * Reads & parses the manifest file.
1488 *
1489 * @param pTask The read task.
1490 * @param hVfsIosMf The I/O stream for the manifest file. The
1491 * reference is always consumed.
1492 * @param pszSubFileNm The manifest filename (no path) for error
1493 * messages and logging.
1494 * @returns COM status code, error info set.
1495 * @throws Nothing
1496 */
1497HRESULT Appliance::i_readManifestFile(TaskOVF *pTask, RTVFSIOSTREAM hVfsIosMf, const char *pszSubFileNm)
1498{
1499 LogFlowFunc(("%s[%s]\n", pTask->locInfo.strPath.c_str(), pszSubFileNm));
1500
1501 /*
1502 * Copy the manifest into a memory backed file so we can later do signature
1503 * validation indepentend of the algorithms used by the signature.
1504 */
1505 int vrc = RTVfsMemorizeIoStreamAsFile(hVfsIosMf, RTFILE_O_READ, &m->hMemFileTheirManifest);
1506 RTVfsIoStrmRelease(hVfsIosMf); /* consumes stream handle. */
1507 if (RT_FAILURE(vrc))
1508 return setErrorVrc(vrc, tr("Error reading the manifest file '%s' for '%s' (%Rrc)"),
1509 pszSubFileNm, pTask->locInfo.strPath.c_str(), vrc);
1510
1511 /*
1512 * Parse the manifest.
1513 */
1514 Assert(m->hTheirManifest == NIL_RTMANIFEST);
1515 vrc = RTManifestCreate(0 /*fFlags*/, &m->hTheirManifest);
1516 AssertStmt(RT_SUCCESS(vrc), Global::vboxStatusCodeToCOM(vrc));
1517
1518 char szErr[256];
1519 RTVFSIOSTREAM hVfsIos = RTVfsFileToIoStream(m->hMemFileTheirManifest);
1520 vrc = RTManifestReadStandardEx(m->hTheirManifest, hVfsIos, szErr, sizeof(szErr));
1521 RTVfsIoStrmRelease(hVfsIos);
1522 if (RT_FAILURE(vrc))
1523 throw setErrorVrc(vrc, tr("Failed to parse manifest file '%s' for '%s' (%Rrc): %s"),
1524 pszSubFileNm, pTask->locInfo.strPath.c_str(), vrc, szErr);
1525
1526 /*
1527 * Check which digest files are used.
1528 * Note! the file could be empty, in which case fDigestTypes is set to 0.
1529 */
1530 vrc = RTManifestQueryAllAttrTypes(m->hTheirManifest, true /*fEntriesOnly*/, &m->fDigestTypes);
1531 AssertRCReturn(vrc, Global::vboxStatusCodeToCOM(vrc));
1532 m->fDeterminedDigestTypes = true;
1533
1534 m->fSha256 = RT_BOOL(m->fDigestTypes & RTMANIFEST_ATTR_SHA256); /** @todo retire this member */
1535 return S_OK;
1536}
1537
1538/**
1539 * Reads the signature & certificate file.
1540 *
1541 * @param pTask The read task.
1542 * @param hVfsIosCert The I/O stream for the signature file. The
1543 * reference is always consumed.
1544 * @param pszSubFileNm The signature filename (no path) for error
1545 * messages and logging. Used to construct
1546 * .mf-file name.
1547 * @returns COM status code, error info set.
1548 * @throws Nothing
1549 */
1550HRESULT Appliance::i_readSignatureFile(TaskOVF *pTask, RTVFSIOSTREAM hVfsIosCert, const char *pszSubFileNm)
1551{
1552 LogFlowFunc(("%s[%s]\n", pTask->locInfo.strPath.c_str(), pszSubFileNm));
1553
1554 /*
1555 * Construct the manifest filename from pszSubFileNm.
1556 */
1557 Utf8Str strManifestName;
1558 try
1559 {
1560 const char *pszSuffix = strrchr(pszSubFileNm, '.');
1561 AssertReturn(pszSuffix, E_FAIL);
1562 strManifestName = Utf8Str(pszSubFileNm, pszSuffix - pszSubFileNm);
1563 strManifestName.append(".mf");
1564 }
1565 catch (...)
1566 {
1567 return E_OUTOFMEMORY;
1568 }
1569
1570 /*
1571 * Copy the manifest into a memory buffer. We'll do the signature processing
1572 * later to not force any specific order in the OVAs or any other archive we
1573 * may be accessing later.
1574 */
1575 void *pvSignature;
1576 size_t cbSignature;
1577 int vrc = RTVfsIoStrmReadAll(hVfsIosCert, &pvSignature, &cbSignature);
1578 RTVfsIoStrmRelease(hVfsIosCert); /* consumes stream handle. */
1579 if (RT_FAILURE(vrc))
1580 return setErrorVrc(vrc, tr("Error reading the signature file '%s' for '%s' (%Rrc)"),
1581 pszSubFileNm, pTask->locInfo.strPath.c_str(), vrc);
1582
1583 /*
1584 * Parse the signing certificate. Unlike the manifest parser we use below,
1585 * this API ignores parse of the file that aren't relevant.
1586 */
1587 RTERRINFOSTATIC StaticErrInfo;
1588 vrc = RTCrX509Certificate_ReadFromBuffer(&m->SignerCert, pvSignature, cbSignature, 0 /*fFlags*/,
1589 &g_RTAsn1DefaultAllocator, RTErrInfoInitStatic(&StaticErrInfo), pszSubFileNm);
1590 HRESULT hrc;
1591 if (RT_SUCCESS(vrc))
1592 {
1593 m->fSignerCertLoaded = true;
1594 m->fCertificateIsSelfSigned = RTCrX509Certificate_IsSelfSigned(&m->SignerCert);
1595
1596 /*
1597 * Find the start of the certificate part of the file, so we can avoid
1598 * upsetting the manifest parser with it.
1599 */
1600 char *pszSplit = (char *)RTCrPemFindFirstSectionInContent(pvSignature, cbSignature,
1601 g_aRTCrX509CertificateMarkers, g_cRTCrX509CertificateMarkers);
1602 if (pszSplit)
1603 while ( pszSplit != (char *)pvSignature
1604 && pszSplit[-1] != '\n'
1605 && pszSplit[-1] != '\r')
1606 pszSplit--;
1607 else
1608 {
1609 AssertLogRelMsgFailed(("Failed to find BEGIN CERTIFICATE markers in '%s'::'%s' - impossible unless it's a DER encoded certificate!",
1610 pTask->locInfo.strPath.c_str(), pszSubFileNm));
1611 pszSplit = (char *)pvSignature + cbSignature;
1612 }
1613 *pszSplit = '\0';
1614
1615 /*
1616 * Now, read the manifest part. We use the IPRT manifest reader here
1617 * to avoid duplicating code and be somewhat flexible wrt the digest
1618 * type choosen by the signer.
1619 */
1620 RTMANIFEST hSignedDigestManifest;
1621 vrc = RTManifestCreate(0 /*fFlags*/, &hSignedDigestManifest);
1622 if (RT_SUCCESS(vrc))
1623 {
1624 RTVFSIOSTREAM hVfsIosTmp;
1625 vrc = RTVfsIoStrmFromBuffer(RTFILE_O_READ, pvSignature, pszSplit - (char *)pvSignature, &hVfsIosTmp);
1626 if (RT_SUCCESS(vrc))
1627 {
1628 vrc = RTManifestReadStandardEx(hSignedDigestManifest, hVfsIosTmp, StaticErrInfo.szMsg, sizeof(StaticErrInfo.szMsg));
1629 RTVfsIoStrmRelease(hVfsIosTmp);
1630 if (RT_SUCCESS(vrc))
1631 {
1632 /*
1633 * Get signed digest, we prefer SHA-2, so explicitly query those first.
1634 */
1635 uint32_t fDigestType;
1636 char szSignedDigest[_8K + 1];
1637 vrc = RTManifestEntryQueryAttr(hSignedDigestManifest, strManifestName.c_str(), NULL,
1638 RTMANIFEST_ATTR_SHA512 | RTMANIFEST_ATTR_SHA256,
1639 szSignedDigest, sizeof(szSignedDigest), &fDigestType);
1640 if (vrc == VERR_MANIFEST_ATTR_TYPE_NOT_FOUND)
1641 vrc = RTManifestEntryQueryAttr(hSignedDigestManifest, strManifestName.c_str(), NULL,
1642 RTMANIFEST_ATTR_ANY, szSignedDigest, sizeof(szSignedDigest), &fDigestType);
1643 if (RT_SUCCESS(vrc))
1644 {
1645 const char *pszSignedDigest = RTStrStrip(szSignedDigest);
1646 size_t cbSignedDigest = strlen(pszSignedDigest) / 2;
1647 uint8_t abSignedDigest[sizeof(szSignedDigest) / 2];
1648 vrc = RTStrConvertHexBytes(szSignedDigest, abSignedDigest, cbSignedDigest, 0 /*fFlags*/);
1649 if (RT_SUCCESS(vrc))
1650 {
1651 /*
1652 * Convert it to RTDIGESTTYPE_XXX and save the binary value for later use.
1653 */
1654 switch (fDigestType)
1655 {
1656 case RTMANIFEST_ATTR_SHA1: m->enmSignedDigestType = RTDIGESTTYPE_SHA1; break;
1657 case RTMANIFEST_ATTR_SHA256: m->enmSignedDigestType = RTDIGESTTYPE_SHA256; break;
1658 case RTMANIFEST_ATTR_SHA512: m->enmSignedDigestType = RTDIGESTTYPE_SHA512; break;
1659 case RTMANIFEST_ATTR_MD5: m->enmSignedDigestType = RTDIGESTTYPE_MD5; break;
1660 default: AssertFailed(); m->enmSignedDigestType = RTDIGESTTYPE_INVALID; break;
1661 }
1662 if (m->enmSignedDigestType != RTDIGESTTYPE_INVALID)
1663 {
1664 m->pbSignedDigest = (uint8_t *)RTMemDup(abSignedDigest, cbSignedDigest);
1665 m->cbSignedDigest = cbSignedDigest;
1666 hrc = S_OK;
1667 }
1668 else
1669 hrc = setError(E_FAIL, tr("Unsupported signed digest type (%#x)"), fDigestType);
1670 }
1671 else
1672 hrc = setErrorVrc(vrc, tr("Error reading signed manifest digest: %Rrc"), vrc);
1673 }
1674 else if (vrc == VERR_NOT_FOUND)
1675 hrc = setErrorVrc(vrc, tr("Could not locate signed digest for '%s' in the cert-file for '%s'"),
1676 strManifestName.c_str(), pTask->locInfo.strPath.c_str());
1677 else
1678 hrc = setErrorVrc(vrc, tr("RTManifestEntryQueryAttr failed unexpectedly: %Rrc"), vrc);
1679 }
1680 else
1681 hrc = setErrorVrc(vrc, tr("Error parsing the .cert-file for '%s': %s"),
1682 pTask->locInfo.strPath.c_str(), StaticErrInfo.szMsg);
1683 }
1684 else
1685 hrc = E_OUTOFMEMORY;
1686 RTManifestRelease(hSignedDigestManifest);
1687 }
1688 else
1689 hrc = E_OUTOFMEMORY;
1690 }
1691 else
1692 hrc = setErrorVrc(vrc, tr("Error reading the signer's certificate from '%s' for '%s' (%Rrc): %s"),
1693 pszSubFileNm, pTask->locInfo.strPath.c_str(), vrc, StaticErrInfo.Core.pszMsg);
1694
1695 RTVfsIoStrmReadAllFree(pvSignature, cbSignature);
1696 LogFlowFunc(("returns %Rhrc (%Rrc)\n", hrc, vrc));
1697 return hrc;
1698}
1699
1700
1701/**
1702 * Does tail processing after the files have been read in.
1703 *
1704 * @param pTask The read task.
1705 * @returns COM status.
1706 * @throws Nothing!
1707 */
1708HRESULT Appliance::i_readTailProcessing(TaskOVF *pTask)
1709{
1710 /*
1711 * Parse and validate the signature file.
1712 *
1713 * The signature file has two parts, manifest part and a PEM encoded
1714 * certificate. The former contains an entry for the manifest file with a
1715 * digest that is encrypted with the certificate in the latter part.
1716 */
1717 if (m->pbSignedDigest)
1718 {
1719 /* Since we're validating the digest of the manifest, there have to be
1720 a manifest. We cannot allow a the manifest to be missing. */
1721 if (m->hMemFileTheirManifest == NIL_RTVFSFILE)
1722 return setError(VBOX_E_FILE_ERROR, tr("Found .cert-file but no .mf-file for '%s'"), pTask->locInfo.strPath.c_str());
1723
1724 /*
1725 * Validate the signed digest.
1726 *
1727 * It's possible we should allow the user to ignore signature
1728 * mismatches, but for now it is a solid show stopper.
1729 */
1730 HRESULT hrc;
1731 RTERRINFOSTATIC StaticErrInfo;
1732
1733 /* Calc the digest of the manifest using the algorithm found above. */
1734 RTCRDIGEST hDigest;
1735 int vrc = RTCrDigestCreateByType(&hDigest, m->enmSignedDigestType);
1736 if (RT_SUCCESS(vrc))
1737 {
1738 vrc = RTCrDigestUpdateFromVfsFile(hDigest, m->hMemFileTheirManifest, true /*fRewindFile*/);
1739 if (RT_SUCCESS(vrc))
1740 {
1741 /* Compare the signed digest with the one we just calculated. (This
1742 API will do the verification twice, once using IPRT's own crypto
1743 and once using OpenSSL. Both must OK it for success.) */
1744 vrc = RTCrPkixPubKeyVerifySignedDigest(&m->SignerCert.TbsCertificate.SubjectPublicKeyInfo.Algorithm.Algorithm,
1745 &m->SignerCert.TbsCertificate.SubjectPublicKeyInfo.Algorithm.Parameters,
1746 &m->SignerCert.TbsCertificate.SubjectPublicKeyInfo.SubjectPublicKey,
1747 m->pbSignedDigest, m->cbSignedDigest, hDigest,
1748 RTErrInfoInitStatic(&StaticErrInfo));
1749 if (RT_SUCCESS(vrc))
1750 {
1751 m->fSignatureValid = true;
1752 hrc = S_OK;
1753 }
1754 else if (vrc == VERR_CR_PKIX_SIGNATURE_MISMATCH)
1755 hrc = setErrorVrc(vrc, tr("The manifest signature does not match"));
1756 else
1757 hrc = setErrorVrc(vrc,
1758 tr("Error validating the manifest signature (%Rrc, %s)"), vrc, StaticErrInfo.Core.pszMsg);
1759 }
1760 else
1761 hrc = setErrorVrc(vrc, tr("RTCrDigestUpdateFromVfsFile failed: %Rrc"), vrc);
1762 RTCrDigestRelease(hDigest);
1763 }
1764 else
1765 hrc = setErrorVrc(vrc, tr("RTCrDigestCreateByType failed: %Rrc"), vrc);
1766
1767 /*
1768 * Validate the certificate.
1769 *
1770 * We don't fail here on if we cannot validate the certificate, we postpone
1771 * that till the import stage, so that we can allow the user to ignore it.
1772 *
1773 * The certificate validity time is deliberately left as warnings as the
1774 * OVF specification does not provision for any timestamping of the
1775 * signature. This is course a security concern, but the whole signing
1776 * of OVFs is currently weirdly trusting (self signed * certs), so this
1777 * is the least of our current problems.
1778 *
1779 * While we try build and verify certificate paths properly, the
1780 * "neighbours" quietly ignores this and seems only to check the signature
1781 * and not whether the certificate is trusted. Also, we don't currently
1782 * complain about self-signed certificates either (ditto "neighbours").
1783 * The OVF creator is also a bit restricted wrt to helping us build the
1784 * path as he cannot supply intermediate certificates. Anyway, we issue
1785 * warnings (goes to /dev/null, am I right?) for self-signed certificates
1786 * and certificates we cannot build and verify a root path for.
1787 *
1788 * (The OVF sillibuggers should've used PKCS#7, CMS or something else
1789 * that's already been standardized instead of combining manifests with
1790 * certificate PEM files in some very restrictive manner! I wonder if
1791 * we could add a PKCS#7 section to the .cert file in addition to the CERT
1792 * and manifest stuff dictated by the standard. Would depend on how others
1793 * deal with it.)
1794 */
1795 Assert(!m->fCertificateValid);
1796 Assert(m->fCertificateMissingPath);
1797 Assert(!m->fCertificateValidTime);
1798 Assert(m->strCertError.isEmpty());
1799 Assert(m->fCertificateIsSelfSigned == RTCrX509Certificate_IsSelfSigned(&m->SignerCert));
1800
1801 HRESULT hrc2 = S_OK;
1802 if (m->fCertificateIsSelfSigned)
1803 {
1804 /*
1805 * It's a self signed certificate. We assume the frontend will
1806 * present this fact to the user and give a choice whether this
1807 * is acceptible. But, first make sure it makes internal sense.
1808 */
1809 m->fCertificateMissingPath = false;
1810 vrc = RTCrX509Certificate_VerifySignatureSelfSigned(&m->SignerCert, RTErrInfoInitStatic(&StaticErrInfo));
1811 if (RT_SUCCESS(vrc))
1812 {
1813 m->fCertificateValid = true;
1814
1815 /* Check whether the certificate is currently valid, just warn if not. */
1816 RTTIMESPEC Now;
1817 if (RTCrX509Validity_IsValidAtTimeSpec(&m->SignerCert.TbsCertificate.Validity, RTTimeNow(&Now)))
1818 {
1819 m->fCertificateValidTime = true;
1820 i_addWarning(tr("A self signed certificate was used to sign '%s'"), pTask->locInfo.strPath.c_str());
1821 }
1822 else
1823 i_addWarning(tr("Self signed certificate used to sign '%s' is not currently valid"),
1824 pTask->locInfo.strPath.c_str());
1825
1826 /* Just warn if it's not a CA. Self-signed certificates are
1827 hardly trustworthy to start with without the user's consent. */
1828 if ( !m->SignerCert.TbsCertificate.T3.pBasicConstraints
1829 || !m->SignerCert.TbsCertificate.T3.pBasicConstraints->CA.fValue)
1830 i_addWarning(tr("Self signed certificate used to sign '%s' is not marked as certificate authority (CA)"),
1831 pTask->locInfo.strPath.c_str());
1832 }
1833 else
1834 {
1835 try { m->strCertError = Utf8StrFmt(tr("Verification of the self signed certificate failed (%Rrc, %s)"),
1836 vrc, StaticErrInfo.Core.pszMsg); }
1837 catch (...) { AssertFailed(); }
1838 i_addWarning(tr("Verification of the self signed certificate used to sign '%s' failed (%Rrc): %s"),
1839 pTask->locInfo.strPath.c_str(), vrc, StaticErrInfo.Core.pszMsg);
1840 }
1841 }
1842 else
1843 {
1844 /*
1845 * The certificate is not self-signed. Use the system certificate
1846 * stores to try build a path that validates successfully.
1847 */
1848 RTCRX509CERTPATHS hCertPaths;
1849 vrc = RTCrX509CertPathsCreate(&hCertPaths, &m->SignerCert);
1850 if (RT_SUCCESS(vrc))
1851 {
1852 /* Get trusted certificates from the system and add them to the path finding mission. */
1853 RTCRSTORE hTrustedCerts;
1854 vrc = RTCrStoreCreateSnapshotOfUserAndSystemTrustedCAsAndCerts(&hTrustedCerts,
1855 RTErrInfoInitStatic(&StaticErrInfo));
1856 if (RT_SUCCESS(vrc))
1857 {
1858 vrc = RTCrX509CertPathsSetTrustedStore(hCertPaths, hTrustedCerts);
1859 if (RT_FAILURE(vrc))
1860 hrc2 = setError(E_FAIL, tr("RTCrX509CertPathsSetTrustedStore failed (%Rrc)"), vrc);
1861 RTCrStoreRelease(hTrustedCerts);
1862 }
1863 else
1864 hrc2 = setError(E_FAIL,
1865 tr("Failed to query trusted CAs and Certificates from the system and for the current user (%Rrc, %s)"),
1866 vrc, StaticErrInfo.Core.pszMsg);
1867
1868 /* Add untrusted intermediate certificates. */
1869 if (RT_SUCCESS(vrc))
1870 {
1871 /// @todo RTCrX509CertPathsSetUntrustedStore(hCertPaths, hAdditionalCerts);
1872 /// By scanning for additional certificates in the .cert file? It would be
1873 /// convenient to be able to supply intermediate certificates for the user,
1874 /// right? Or would that be unacceptable as it may weaken security?
1875 ///
1876 /// Anyway, we should look for intermediate certificates on the system, at
1877 /// least.
1878 }
1879 if (RT_SUCCESS(vrc))
1880 {
1881 /*
1882 * Do the building and verification of certificate paths.
1883 */
1884 vrc = RTCrX509CertPathsBuild(hCertPaths, RTErrInfoInitStatic(&StaticErrInfo));
1885 if (RT_SUCCESS(vrc))
1886 {
1887 vrc = RTCrX509CertPathsValidateAll(hCertPaths, NULL, RTErrInfoInitStatic(&StaticErrInfo));
1888 if (RT_SUCCESS(vrc))
1889 {
1890 /*
1891 * Mark the certificate as good.
1892 */
1893 /** @todo check the certificate purpose? If so, share with self-signed. */
1894 m->fCertificateValid = true;
1895 m->fCertificateMissingPath = false;
1896
1897 /*
1898 * We add a warning if the certificate path isn't valid at the current
1899 * time. Since the time is only considered during path validation and we
1900 * can repeat the validation process (but not building), it's easy to check.
1901 */
1902 RTTIMESPEC Now;
1903 vrc = RTCrX509CertPathsSetValidTimeSpec(hCertPaths, RTTimeNow(&Now));
1904 if (RT_SUCCESS(vrc))
1905 {
1906 vrc = RTCrX509CertPathsValidateAll(hCertPaths, NULL, RTErrInfoInitStatic(&StaticErrInfo));
1907 if (RT_SUCCESS(vrc))
1908 m->fCertificateValidTime = true;
1909 else
1910 i_addWarning(tr("The certificate used to sign '%s' (or a certificate in the path) is not currently valid (%Rrc)"),
1911 pTask->locInfo.strPath.c_str(), vrc);
1912 }
1913 else
1914 hrc2 = setErrorVrc(vrc, "RTCrX509CertPathsSetValidTimeSpec failed: %Rrc", vrc);
1915 }
1916 else if (vrc == VERR_CR_X509_CPV_NO_TRUSTED_PATHS)
1917 {
1918 m->fCertificateValid = true;
1919 i_addWarning(tr("No trusted certificate paths"));
1920
1921 /* Add another warning if the pathless certificate is not valid at present. */
1922 RTTIMESPEC Now;
1923 if (RTCrX509Validity_IsValidAtTimeSpec(&m->SignerCert.TbsCertificate.Validity, RTTimeNow(&Now)))
1924 m->fCertificateValidTime = true;
1925 else
1926 i_addWarning(tr("The certificate used to sign '%s' is not currently valid"),
1927 pTask->locInfo.strPath.c_str());
1928 }
1929 else
1930 hrc2 = setError(E_FAIL, tr("Certificate path validation failed (%Rrc, %s)"),
1931 vrc, StaticErrInfo.Core.pszMsg);
1932 }
1933 else
1934 hrc2 = setError(E_FAIL, tr("Certificate path building failed (%Rrc, %s)"),
1935 vrc, StaticErrInfo.Core.pszMsg);
1936 }
1937 RTCrX509CertPathsRelease(hCertPaths);
1938 }
1939 else
1940 hrc2 = setErrorVrc(vrc, tr("RTCrX509CertPathsCreate failed: %Rrc"), vrc);
1941 }
1942
1943 /* Merge statuses from signature and certificate validation, prefering the signature one. */
1944 if (SUCCEEDED(hrc) && FAILED(hrc2))
1945 hrc = hrc2;
1946 if (FAILED(hrc))
1947 return hrc;
1948 }
1949
1950 /** @todo provide details about the signatory, signature, etc. */
1951
1952 /*
1953 * If there is a manifest, check that the OVF digest matches up (if present).
1954 */
1955
1956 NOREF(pTask);
1957 return S_OK;
1958}
1959
1960
1961
1962/*******************************************************************************
1963 * Import stuff
1964 ******************************************************************************/
1965
1966/**
1967 * Implementation for importing OVF data into VirtualBox. This starts a new thread which will call
1968 * Appliance::taskThreadImportOrExport().
1969 *
1970 * This creates one or more new machines according to the VirtualSystemScription instances created by
1971 * Appliance::Interpret().
1972 *
1973 * This is in a separate private method because it is used from one location:
1974 *
1975 * 1) from the public Appliance::ImportMachines().
1976 *
1977 * @param aLocInfo
1978 * @param aProgress
1979 * @return
1980 */
1981HRESULT Appliance::i_importImpl(const LocationInfo &locInfo,
1982 ComObjPtr<Progress> &progress)
1983{
1984 HRESULT rc = S_OK;
1985
1986 SetUpProgressMode mode;
1987 if (locInfo.storageType == VFSType_File)
1988 mode = ImportFile;
1989 else
1990 mode = ImportS3;
1991
1992 rc = i_setUpProgress(progress,
1993 BstrFmt(tr("Importing appliance '%s'"), locInfo.strPath.c_str()),
1994 mode);
1995 if (FAILED(rc)) throw rc;
1996
1997 /* Initialize our worker task */
1998 TaskOVF* task = NULL;
1999 try
2000 {
2001 task = new TaskOVF(this, TaskOVF::Import, locInfo, progress);
2002 }
2003 catch(...)
2004 {
2005 delete task;
2006 throw rc = setError(VBOX_E_OBJECT_NOT_FOUND,
2007 tr("Could not create TaskOVF object for importing OVF data into VirtualBox"));
2008 }
2009
2010 rc = task->createThread();
2011 if (FAILED(rc)) throw rc;
2012
2013 return rc;
2014}
2015
2016/**
2017 * Actual worker code for importing OVF data into VirtualBox.
2018 *
2019 * This is called from Appliance::taskThreadImportOrExport() and therefore runs
2020 * on the OVF import worker thread. This creates one or more new machines
2021 * according to the VirtualSystemScription instances created by
2022 * Appliance::Interpret().
2023 *
2024 * This runs in two contexts:
2025 *
2026 * 1) in a first worker thread; in that case, Appliance::ImportMachines() called
2027 * Appliance::i_importImpl();
2028 *
2029 * 2) in a second worker thread; in that case, Appliance::ImportMachines()
2030 * called Appliance::i_importImpl(), which called Appliance::i_importFSOVA(),
2031 * which called Appliance::i_importImpl(), which then called this again.
2032 *
2033 * @param pTask The OVF task data.
2034 * @return COM status code.
2035 */
2036HRESULT Appliance::i_importFS(TaskOVF *pTask)
2037{
2038 LogFlowFuncEnter();
2039 LogFlowFunc(("Appliance %p\n", this));
2040
2041 /* Change the appliance state so we can safely leave the lock while doing
2042 * time-consuming disk imports; also the below method calls do all kinds of
2043 * locking which conflicts with the appliance object lock. */
2044 AutoWriteLock writeLock(this COMMA_LOCKVAL_SRC_POS);
2045 /* Check if the appliance is currently busy. */
2046 if (!i_isApplianceIdle())
2047 return E_ACCESSDENIED;
2048 /* Set the internal state to importing. */
2049 m->state = Data::ApplianceImporting;
2050
2051 HRESULT rc = S_OK;
2052
2053 /* Clear the list of imported machines, if any */
2054 m->llGuidsMachinesCreated.clear();
2055
2056 if (pTask->locInfo.strPath.endsWith(".ovf", Utf8Str::CaseInsensitive))
2057 rc = i_importFSOVF(pTask, writeLock);
2058 else
2059 rc = i_importFSOVA(pTask, writeLock);
2060 if (FAILED(rc))
2061 {
2062 /* With _whatever_ error we've had, do a complete roll-back of
2063 * machines and disks we've created */
2064 writeLock.release();
2065 ErrorInfoKeeper eik;
2066 for (list<Guid>::iterator itID = m->llGuidsMachinesCreated.begin();
2067 itID != m->llGuidsMachinesCreated.end();
2068 ++itID)
2069 {
2070 Guid guid = *itID;
2071 Bstr bstrGuid = guid.toUtf16();
2072 ComPtr<IMachine> failedMachine;
2073 HRESULT rc2 = mVirtualBox->FindMachine(bstrGuid.raw(), failedMachine.asOutParam());
2074 if (SUCCEEDED(rc2))
2075 {
2076 SafeIfaceArray<IMedium> aMedia;
2077 rc2 = failedMachine->Unregister(CleanupMode_DetachAllReturnHardDisksOnly, ComSafeArrayAsOutParam(aMedia));
2078 ComPtr<IProgress> pProgress2;
2079 rc2 = failedMachine->DeleteConfig(ComSafeArrayAsInParam(aMedia), pProgress2.asOutParam());
2080 pProgress2->WaitForCompletion(-1);
2081 }
2082 }
2083 writeLock.acquire();
2084 }
2085
2086 /* Reset the state so others can call methods again */
2087 m->state = Data::ApplianceIdle;
2088
2089 LogFlowFunc(("rc=%Rhrc\n", rc));
2090 LogFlowFuncLeave();
2091 return rc;
2092}
2093
2094HRESULT Appliance::i_importFSOVF(TaskOVF *pTask, AutoWriteLockBase &rWriteLock)
2095{
2096 return i_importDoIt(pTask, rWriteLock);
2097}
2098
2099HRESULT Appliance::i_importFSOVA(TaskOVF *pTask, AutoWriteLockBase &rWriteLock)
2100{
2101 LogFlowFuncEnter();
2102
2103 /*
2104 * Open the tar file as file stream.
2105 */
2106 RTVFSIOSTREAM hVfsIosOva;
2107 int vrc = RTVfsIoStrmOpenNormal(pTask->locInfo.strPath.c_str(),
2108 RTFILE_O_READ | RTFILE_O_DENY_NONE | RTFILE_O_OPEN, &hVfsIosOva);
2109 if (RT_FAILURE(vrc))
2110 return setErrorVrc(vrc, tr("Error opening the OVA file '%s' (%Rrc)"), pTask->locInfo.strPath.c_str(), vrc);
2111
2112 RTVFSFSSTREAM hVfsFssOva;
2113 vrc = RTZipTarFsStreamFromIoStream(hVfsIosOva, 0 /*fFlags*/, &hVfsFssOva);
2114 RTVfsIoStrmRelease(hVfsIosOva);
2115 if (RT_FAILURE(vrc))
2116 return setErrorVrc(vrc, tr("Error reading the OVA file '%s' (%Rrc)"), pTask->locInfo.strPath.c_str(), vrc);
2117
2118 /*
2119 * Join paths with the i_importFSOVF code.
2120 *
2121 * Note! We don't need to skip the OVF, manifest or signature files, as the
2122 * i_importMachineGeneric, i_importVBoxMachine and i_importOpenSourceFile
2123 * code will deal with this (as there could be other files in the OVA
2124 * that we don't process, like 'de-DE-resources.xml' in EXAMPLE 1,
2125 * Appendix D.1, OVF v2.1.0).
2126 */
2127 HRESULT hrc = i_importDoIt(pTask, rWriteLock, hVfsFssOva);
2128
2129 RTVfsFsStrmRelease(hVfsFssOva);
2130
2131 LogFlowFunc(("returns %Rhrc\n", hrc));
2132 return hrc;
2133}
2134
2135/**
2136 * Does the actual importing after the caller has made the source accessible.
2137 *
2138 * @param pTask The import task.
2139 * @param rWriteLock The write lock the caller's caller is holding,
2140 * will be released for some reason.
2141 * @param hVfsFssOva The file system stream if OVA, NIL if not.
2142 * @returns COM status code.
2143 * @throws Nothing.
2144 */
2145HRESULT Appliance::i_importDoIt(TaskOVF *pTask, AutoWriteLockBase &rWriteLock, RTVFSFSSTREAM hVfsFssOva /*= NIL_RTVFSFSSTREAM*/)
2146{
2147 rWriteLock.release();
2148
2149 HRESULT hrc = E_FAIL;
2150 try
2151 {
2152 /*
2153 * Create the import stack for the rollback on errors.
2154 */
2155 ImportStack stack(pTask->locInfo, m->pReader->m_mapDisks, pTask->pProgress, hVfsFssOva);
2156
2157 try
2158 {
2159 /* Do the importing. */
2160 i_importMachines(stack);
2161
2162 /* We should've processed all the files now, so compare. */
2163 hrc = i_verifyManifestFile(stack);
2164 }
2165 catch (HRESULT hrcXcpt)
2166 {
2167 hrc = hrcXcpt;
2168 }
2169 catch (...)
2170 {
2171 AssertFailed();
2172 hrc = E_FAIL;
2173 }
2174 if (FAILED(hrc))
2175 {
2176 /*
2177 * Restoring original UUID from OVF description file.
2178 * During import VBox creates new UUIDs for imported images and
2179 * assigns them to the images. In case of failure we have to restore
2180 * the original UUIDs because those new UUIDs are obsolete now and
2181 * won't be used anymore.
2182 */
2183 ErrorInfoKeeper eik; /* paranoia */
2184 list< ComObjPtr<VirtualSystemDescription> >::const_iterator itvsd;
2185 /* Iterate through all virtual systems of that appliance */
2186 for (itvsd = m->virtualSystemDescriptions.begin();
2187 itvsd != m->virtualSystemDescriptions.end();
2188 ++itvsd)
2189 {
2190 ComObjPtr<VirtualSystemDescription> vsdescThis = (*itvsd);
2191 settings::MachineConfigFile *pConfig = vsdescThis->m->pConfig;
2192 if(vsdescThis->m->pConfig!=NULL)
2193 stack.restoreOriginalUUIDOfAttachedDevice(pConfig);
2194 }
2195 }
2196 }
2197 catch (...)
2198 {
2199 hrc = E_FAIL;
2200 AssertFailed();
2201 }
2202
2203 rWriteLock.acquire();
2204 return hrc;
2205}
2206
2207/**
2208 * Undocumented, you figure it from the name.
2209 *
2210 * @returns Undocumented
2211 * @param stack Undocumented.
2212 */
2213HRESULT Appliance::i_verifyManifestFile(ImportStack &stack)
2214{
2215 LogFlowThisFuncEnter();
2216 HRESULT hrc;
2217 int vrc;
2218
2219 /*
2220 * No manifest is fine, it always matches.
2221 */
2222 if (m->hTheirManifest == NIL_RTMANIFEST)
2223 hrc = S_OK;
2224 else
2225 {
2226 /*
2227 * Hack: If the manifest we just read doesn't have a digest for the OVF, copy
2228 * it from the manifest we got from the caller.
2229 * @bugref{6022#c119}
2230 */
2231 if ( !RTManifestEntryExists(m->hTheirManifest, m->strOvfManifestEntry.c_str())
2232 && RTManifestEntryExists(m->hOurManifest, m->strOvfManifestEntry.c_str()) )
2233 {
2234 uint32_t fType = 0;
2235 char szDigest[512 + 1];
2236 vrc = RTManifestEntryQueryAttr(m->hOurManifest, m->strOvfManifestEntry.c_str(), NULL, RTMANIFEST_ATTR_ANY,
2237 szDigest, sizeof(szDigest), &fType);
2238 if (RT_SUCCESS(vrc))
2239 vrc = RTManifestEntrySetAttr(m->hTheirManifest, m->strOvfManifestEntry.c_str(),
2240 NULL /*pszAttr*/, szDigest, fType);
2241 if (RT_FAILURE(vrc))
2242 return setError(VBOX_E_IPRT_ERROR, tr("Error fudging missing OVF digest in manifest: %Rrc"), vrc);
2243 }
2244
2245 /*
2246 * Compare with the digests we've created while read/processing the import.
2247 *
2248 * We specify the RTMANIFEST_EQUALS_IGN_MISSING_ATTRS to ignore attributes
2249 * (SHA1, SHA256, etc) that are only present in one of the manifests, as long
2250 * as each entry has at least one common attribute that we can check. This
2251 * is important for the OVF in OVAs, for which we generates several digests
2252 * since we don't know which are actually used in the manifest (OVF comes
2253 * first in an OVA, then manifest).
2254 */
2255 char szErr[256];
2256 vrc = RTManifestEqualsEx(m->hTheirManifest, m->hOurManifest, NULL /*papszIgnoreEntries*/,
2257 NULL /*papszIgnoreAttrs*/, RTMANIFEST_EQUALS_IGN_MISSING_ATTRS, szErr, sizeof(szErr));
2258 if (RT_SUCCESS(vrc))
2259 hrc = S_OK;
2260 else
2261 hrc = setErrorVrc(vrc, tr("Digest mismatch (%Rrc): %s"), vrc, szErr);
2262 }
2263
2264 NOREF(stack);
2265 LogFlowThisFunc(("returns %Rhrc\n", hrc));
2266 return hrc;
2267}
2268
2269/**
2270 * Helper that converts VirtualSystem attachment values into VirtualBox attachment values.
2271 * Throws HRESULT values on errors!
2272 *
2273 * @param hdc in: the HardDiskController structure to attach to.
2274 * @param ulAddressOnParent in: the AddressOnParent parameter from OVF.
2275 * @param controllerType out: the name of the hard disk controller to attach to (e.g. "IDE Controller").
2276 * @param lControllerPort out: the channel (controller port) of the controller to attach to.
2277 * @param lDevice out: the device number to attach to.
2278 */
2279void Appliance::i_convertDiskAttachmentValues(const ovf::HardDiskController &hdc,
2280 uint32_t ulAddressOnParent,
2281 Bstr &controllerType,
2282 int32_t &lControllerPort,
2283 int32_t &lDevice)
2284{
2285 Log(("Appliance::i_convertDiskAttachmentValues: hdc.system=%d, hdc.fPrimary=%d, ulAddressOnParent=%d\n",
2286 hdc.system,
2287 hdc.fPrimary,
2288 ulAddressOnParent));
2289
2290 switch (hdc.system)
2291 {
2292 case ovf::HardDiskController::IDE:
2293 // For the IDE bus, the port parameter can be either 0 or 1, to specify the primary
2294 // or secondary IDE controller, respectively. For the primary controller of the IDE bus,
2295 // the device number can be either 0 or 1, to specify the master or the slave device,
2296 // respectively. For the secondary IDE controller, the device number is always 1 because
2297 // the master device is reserved for the CD-ROM drive.
2298 controllerType = Bstr("IDE Controller");
2299 switch (ulAddressOnParent)
2300 {
2301 case 0: // master
2302 if (!hdc.fPrimary)
2303 {
2304 // secondary master
2305 lControllerPort = (long)1;
2306 lDevice = (long)0;
2307 }
2308 else // primary master
2309 {
2310 lControllerPort = (long)0;
2311 lDevice = (long)0;
2312 }
2313 break;
2314
2315 case 1: // slave
2316 if (!hdc.fPrimary)
2317 {
2318 // secondary slave
2319 lControllerPort = (long)1;
2320 lDevice = (long)1;
2321 }
2322 else // primary slave
2323 {
2324 lControllerPort = (long)0;
2325 lDevice = (long)1;
2326 }
2327 break;
2328
2329 // used by older VBox exports
2330 case 2: // interpret this as secondary master
2331 lControllerPort = (long)1;
2332 lDevice = (long)0;
2333 break;
2334
2335 // used by older VBox exports
2336 case 3: // interpret this as secondary slave
2337 lControllerPort = (long)1;
2338 lDevice = (long)1;
2339 break;
2340
2341 default:
2342 throw setError(VBOX_E_NOT_SUPPORTED,
2343 tr("Invalid channel %RI16 specified; IDE controllers support only 0, 1 or 2"),
2344 ulAddressOnParent);
2345 break;
2346 }
2347 break;
2348
2349 case ovf::HardDiskController::SATA:
2350 controllerType = Bstr("SATA Controller");
2351 lControllerPort = (long)ulAddressOnParent;
2352 lDevice = (long)0;
2353 break;
2354
2355 case ovf::HardDiskController::SCSI:
2356 {
2357 if(hdc.strControllerType.compare("lsilogicsas")==0)
2358 controllerType = Bstr("SAS Controller");
2359 else
2360 controllerType = Bstr("SCSI Controller");
2361 lControllerPort = (long)ulAddressOnParent;
2362 lDevice = (long)0;
2363 }
2364 break;
2365
2366 default: break;
2367 }
2368
2369 Log(("=> lControllerPort=%d, lDevice=%d\n", lControllerPort, lDevice));
2370}
2371
2372/**
2373 * Imports one disk image.
2374 *
2375 * This is common code shared between
2376 * -- i_importMachineGeneric() for the OVF case; in that case the information comes from
2377 * the OVF virtual systems;
2378 * -- i_importVBoxMachine(); in that case, the information comes from the <vbox:Machine>
2379 * tag.
2380 *
2381 * Both ways of describing machines use the OVF disk references section, so in both cases
2382 * the caller needs to pass in the ovf::DiskImage structure from ovfreader.cpp.
2383 *
2384 * As a result, in both cases, if di.strHref is empty, we create a new disk as per the OVF
2385 * spec, even though this cannot really happen in the vbox:Machine case since such data
2386 * would never have been exported.
2387 *
2388 * This advances stack.pProgress by one operation with the disk's weight.
2389 *
2390 * @param di ovfreader.cpp structure describing the disk image from the OVF that is to be imported
2391 * @param strTargetPath Where to create the target image.
2392 * @param pTargetHD out: The newly created target disk. This also gets pushed on stack.llHardDisksCreated for cleanup.
2393 * @param stack
2394 */
2395void Appliance::i_importOneDiskImage(const ovf::DiskImage &di,
2396 Utf8Str *pStrDstPath,
2397 ComObjPtr<Medium> &pTargetHD,
2398 ImportStack &stack)
2399{
2400 ComObjPtr<Progress> pProgress;
2401 pProgress.createObject();
2402 HRESULT rc = pProgress->init(mVirtualBox,
2403 static_cast<IAppliance*>(this),
2404 BstrFmt(tr("Creating medium '%s'"),
2405 pStrDstPath->c_str()).raw(),
2406 TRUE);
2407 if (FAILED(rc)) throw rc;
2408
2409 /* Get the system properties. */
2410 SystemProperties *pSysProps = mVirtualBox->i_getSystemProperties();
2411
2412 /* Keep the source file ref handy for later. */
2413 const Utf8Str &strSourceOVF = di.strHref;
2414
2415 /* Construct source file path */
2416 Utf8Str strSrcFilePath;
2417 if (stack.hVfsFssOva != NIL_RTVFSFSSTREAM)
2418 strSrcFilePath = strSourceOVF;
2419 else
2420 {
2421 strSrcFilePath = stack.strSourceDir;
2422 strSrcFilePath.append(RTPATH_SLASH_STR);
2423 strSrcFilePath.append(strSourceOVF);
2424 }
2425
2426 /* First of all check if the path is an UUID. If so, the user like to
2427 * import the disk into an existing path. This is useful for iSCSI for
2428 * example. */
2429 RTUUID uuid;
2430 int vrc = RTUuidFromStr(&uuid, pStrDstPath->c_str());
2431 if (vrc == VINF_SUCCESS)
2432 {
2433 rc = mVirtualBox->i_findHardDiskById(Guid(uuid), true, &pTargetHD);
2434 if (FAILED(rc)) throw rc;
2435 }
2436 else
2437 {
2438 RTVFSIOSTREAM hVfsIosSrc = NIL_RTVFSIOSTREAM;
2439
2440 /* check read file to GZIP compression */
2441 bool const fGzipped = di.strCompression.compare("gzip",Utf8Str::CaseInsensitive) == 0;
2442 Utf8Str strDeleteTemp;
2443 try
2444 {
2445 Utf8Str strTrgFormat = "VMDK";
2446 ComObjPtr<MediumFormat> trgFormat;
2447 Bstr bstrFormatName;
2448 ULONG lCabs = 0;
2449
2450 char *pszSuff = RTPathSuffix(pStrDstPath->c_str());
2451 if (pszSuff != NULL)
2452 {
2453 /*
2454 * Figure out which format the user like to have. Default is VMDK
2455 * or it can be VDI if according command-line option is set
2456 */
2457
2458 /*
2459 * We need a proper target format
2460 * if target format has been changed by user via GUI import wizard
2461 * or via VBoxManage import command (option --importtovdi)
2462 * then we need properly process such format like ISO
2463 * Because there is no conversion ISO to VDI
2464 */
2465 trgFormat = pSysProps->i_mediumFormatFromExtension(++pszSuff);
2466 if (trgFormat.isNull())
2467 throw setError(E_FAIL, tr("Unsupported medium format for disk image '%s'"), di.strHref.c_str());
2468
2469 rc = trgFormat->COMGETTER(Name)(bstrFormatName.asOutParam());
2470 if (FAILED(rc)) throw rc;
2471
2472 strTrgFormat = Utf8Str(bstrFormatName);
2473
2474 if ( m->optListImport.contains(ImportOptions_ImportToVDI)
2475 && strTrgFormat.compare("RAW", Utf8Str::CaseInsensitive) != 0)
2476 {
2477 /* change the target extension */
2478 strTrgFormat = "vdi";
2479 trgFormat = pSysProps->i_mediumFormatFromExtension(strTrgFormat);
2480 *pStrDstPath = pStrDstPath->stripSuffix();
2481 *pStrDstPath = pStrDstPath->append(".");
2482 *pStrDstPath = pStrDstPath->append(strTrgFormat.c_str());
2483 }
2484
2485 /* Check the capabilities. We need create capabilities. */
2486 lCabs = 0;
2487 com::SafeArray <MediumFormatCapabilities_T> mediumFormatCap;
2488 rc = trgFormat->COMGETTER(Capabilities)(ComSafeArrayAsOutParam(mediumFormatCap));
2489
2490 if (FAILED(rc))
2491 throw rc;
2492
2493 for (ULONG j = 0; j < mediumFormatCap.size(); j++)
2494 lCabs |= mediumFormatCap[j];
2495
2496 if ( !(lCabs & MediumFormatCapabilities_CreateFixed)
2497 && !(lCabs & MediumFormatCapabilities_CreateDynamic) )
2498 throw setError(VBOX_E_NOT_SUPPORTED,
2499 tr("Could not find a valid medium format for the target disk '%s'"),
2500 pStrDstPath->c_str());
2501 }
2502 else
2503 {
2504 throw setError(VBOX_E_FILE_ERROR,
2505 tr("The target disk '%s' has no extension "),
2506 pStrDstPath->c_str(), VERR_INVALID_NAME);
2507 }
2508
2509 /* Create an IMedium object. */
2510 pTargetHD.createObject();
2511
2512 /*CD/DVD case*/
2513 if (strTrgFormat.compare("RAW", Utf8Str::CaseInsensitive) == 0)
2514 {
2515 try
2516 {
2517 if (fGzipped)
2518 i_importDecompressFile(stack, strSrcFilePath, *pStrDstPath, strSourceOVF.c_str());
2519 else
2520 i_importCopyFile(stack, strSrcFilePath, *pStrDstPath, strSourceOVF.c_str());
2521 }
2522 catch (HRESULT /*arc*/)
2523 {
2524 throw;
2525 }
2526
2527 /* Advance to the next operation. */
2528 /* operation's weight, as set up with the IProgress originally */
2529 stack.pProgress->SetNextOperation(BstrFmt(tr("Importing virtual disk image '%s'"),
2530 RTPathFilename(strSourceOVF.c_str())).raw(),
2531 di.ulSuggestedSizeMB);
2532 }
2533 else/* HDD case*/
2534 {
2535 rc = pTargetHD->init(mVirtualBox,
2536 strTrgFormat,
2537 *pStrDstPath,
2538 Guid::Empty /* media registry: none yet */,
2539 DeviceType_HardDisk);
2540 if (FAILED(rc)) throw rc;
2541
2542 /* Now create an empty hard disk. */
2543 rc = mVirtualBox->CreateMedium(Bstr(strTrgFormat).raw(),
2544 Bstr(*pStrDstPath).raw(),
2545 AccessMode_ReadWrite, DeviceType_HardDisk,
2546 ComPtr<IMedium>(pTargetHD).asOutParam());
2547 if (FAILED(rc)) throw rc;
2548
2549 /* If strHref is empty we have to create a new file. */
2550 if (strSourceOVF.isEmpty())
2551 {
2552 com::SafeArray<MediumVariant_T> mediumVariant;
2553 mediumVariant.push_back(MediumVariant_Standard);
2554
2555 /* Kick of the creation of a dynamic growing disk image with the given capacity. */
2556 rc = pTargetHD->CreateBaseStorage(di.iCapacity / _1M,
2557 ComSafeArrayAsInParam(mediumVariant),
2558 ComPtr<IProgress>(pProgress).asOutParam());
2559 if (FAILED(rc)) throw rc;
2560
2561 /* Advance to the next operation. */
2562 /* operation's weight, as set up with the IProgress originally */
2563 stack.pProgress->SetNextOperation(BstrFmt(tr("Creating disk image '%s'"),
2564 pStrDstPath->c_str()).raw(),
2565 di.ulSuggestedSizeMB);
2566 }
2567 else
2568 {
2569 /* We need a proper source format description */
2570 /* Which format to use? */
2571 ComObjPtr<MediumFormat> srcFormat;
2572 rc = i_findMediumFormatFromDiskImage(di, srcFormat);
2573 if (FAILED(rc))
2574 throw setError(VBOX_E_NOT_SUPPORTED,
2575 tr("Could not find a valid medium format for the source disk '%s' "
2576 "Check correctness of the image format URL in the OVF description file "
2577 "or extension of the image"),
2578 RTPathFilename(strSourceOVF.c_str()));
2579
2580 /* If gzipped, decompress the GZIP file and save a new file in the target path */
2581 if (fGzipped)
2582 {
2583 Utf8Str strTargetFilePath(*pStrDstPath);
2584 strTargetFilePath.stripFilename();
2585 strTargetFilePath.append(RTPATH_SLASH_STR);
2586 strTargetFilePath.append("temp_");
2587 strTargetFilePath.append(RTPathFilename(strSrcFilePath.c_str()));
2588 strDeleteTemp = strTargetFilePath;
2589
2590 i_importDecompressFile(stack, strSrcFilePath, strTargetFilePath, strSourceOVF.c_str());
2591
2592 /* Correct the source and the target with the actual values */
2593 strSrcFilePath = strTargetFilePath;
2594
2595 /* Open the new source file. */
2596 vrc = RTVfsIoStrmOpenNormal(strSrcFilePath.c_str(), RTFILE_O_READ | RTFILE_O_DENY_NONE | RTFILE_O_OPEN,
2597 &hVfsIosSrc);
2598 if (RT_FAILURE(vrc))
2599 throw setErrorVrc(vrc, tr("Error opening decompressed image file '%s' (%Rrc)"),
2600 strSrcFilePath.c_str(), vrc);
2601 }
2602 else
2603 hVfsIosSrc = i_importOpenSourceFile(stack, strSrcFilePath, strSourceOVF.c_str());
2604
2605 /* Add a read ahead thread to try speed things up with concurrent reads and
2606 writes going on in different threads. */
2607 RTVFSIOSTREAM hVfsIosReadAhead;
2608 vrc = RTVfsCreateReadAheadForIoStream(hVfsIosSrc, 0 /*fFlags*/, 0 /*cBuffers=default*/,
2609 0 /*cbBuffers=default*/, &hVfsIosReadAhead);
2610 RTVfsIoStrmRelease(hVfsIosSrc);
2611 if (RT_FAILURE(vrc))
2612 throw setErrorVrc(vrc, tr("Error initializing read ahead thread for '%s' (%Rrc)"),
2613 strSrcFilePath.c_str(), vrc);
2614
2615 /* Start the source image cloning operation. */
2616 ComObjPtr<Medium> nullParent;
2617 rc = pTargetHD->i_importFile(strSrcFilePath.c_str(),
2618 srcFormat,
2619 MediumVariant_Standard,
2620 hVfsIosReadAhead,
2621 nullParent,
2622 pProgress);
2623 RTVfsIoStrmRelease(hVfsIosReadAhead);
2624 hVfsIosSrc = NIL_RTVFSIOSTREAM;
2625 if (FAILED(rc))
2626 throw rc;
2627
2628 /* Advance to the next operation. */
2629 /* operation's weight, as set up with the IProgress originally */
2630 stack.pProgress->SetNextOperation(BstrFmt(tr("Importing virtual disk image '%s'"),
2631 RTPathFilename(strSourceOVF.c_str())).raw(),
2632 di.ulSuggestedSizeMB);
2633 }
2634
2635 /* Now wait for the background disk operation to complete; this throws
2636 * HRESULTs on error. */
2637 ComPtr<IProgress> pp(pProgress);
2638 i_waitForAsyncProgress(stack.pProgress, pp);
2639 }
2640 }
2641 catch (...)
2642 {
2643 if (strDeleteTemp.isNotEmpty())
2644 RTFileDelete(strDeleteTemp.c_str());
2645 throw;
2646 }
2647
2648 /* Make sure the source file is closed. */
2649 if (hVfsIosSrc != NIL_RTVFSIOSTREAM)
2650 RTVfsIoStrmRelease(hVfsIosSrc);
2651
2652 /*
2653 * Delete the temp gunzip result, if any.
2654 */
2655 if (strDeleteTemp.isNotEmpty())
2656 {
2657 vrc = RTFileDelete(strSrcFilePath.c_str());
2658 if (RT_FAILURE(vrc))
2659 setWarning(VBOX_E_FILE_ERROR,
2660 tr("Failed to delete the temporary file '%s' (%Rrc)"), strSrcFilePath.c_str(), vrc);
2661 }
2662 }
2663}
2664
2665/**
2666 * Imports one OVF virtual system (described by the given ovf::VirtualSystem and VirtualSystemDescription)
2667 * into VirtualBox by creating an IMachine instance, which is returned.
2668 *
2669 * This throws HRESULT error codes for anything that goes wrong, in which case the caller must clean
2670 * up any leftovers from this function. For this, the given ImportStack instance has received information
2671 * about what needs cleaning up (to support rollback).
2672 *
2673 * @param vsysThis OVF virtual system (machine) to import.
2674 * @param vsdescThis Matching virtual system description (machine) to import.
2675 * @param pNewMachine out: Newly created machine.
2676 * @param stack Cleanup stack for when this throws.
2677 */
2678void Appliance::i_importMachineGeneric(const ovf::VirtualSystem &vsysThis,
2679 ComObjPtr<VirtualSystemDescription> &vsdescThis,
2680 ComPtr<IMachine> &pNewMachine,
2681 ImportStack &stack)
2682{
2683 LogFlowFuncEnter();
2684 HRESULT rc;
2685
2686 // Get the instance of IGuestOSType which matches our string guest OS type so we
2687 // can use recommended defaults for the new machine where OVF doesn't provide any
2688 ComPtr<IGuestOSType> osType;
2689 rc = mVirtualBox->GetGuestOSType(Bstr(stack.strOsTypeVBox).raw(), osType.asOutParam());
2690 if (FAILED(rc)) throw rc;
2691
2692 /* Create the machine */
2693 SafeArray<BSTR> groups; /* no groups */
2694 rc = mVirtualBox->CreateMachine(NULL, /* machine name: use default */
2695 Bstr(stack.strNameVBox).raw(),
2696 ComSafeArrayAsInParam(groups),
2697 Bstr(stack.strOsTypeVBox).raw(),
2698 NULL, /* aCreateFlags */
2699 pNewMachine.asOutParam());
2700 if (FAILED(rc)) throw rc;
2701
2702 // set the description
2703 if (!stack.strDescription.isEmpty())
2704 {
2705 rc = pNewMachine->COMSETTER(Description)(Bstr(stack.strDescription).raw());
2706 if (FAILED(rc)) throw rc;
2707 }
2708
2709 // CPU count
2710 rc = pNewMachine->COMSETTER(CPUCount)(stack.cCPUs);
2711 if (FAILED(rc)) throw rc;
2712
2713 if (stack.fForceHWVirt)
2714 {
2715 rc = pNewMachine->SetHWVirtExProperty(HWVirtExPropertyType_Enabled, TRUE);
2716 if (FAILED(rc)) throw rc;
2717 }
2718
2719 // RAM
2720 rc = pNewMachine->COMSETTER(MemorySize)(stack.ulMemorySizeMB);
2721 if (FAILED(rc)) throw rc;
2722
2723 /* VRAM */
2724 /* Get the recommended VRAM for this guest OS type */
2725 ULONG vramVBox;
2726 rc = osType->COMGETTER(RecommendedVRAM)(&vramVBox);
2727 if (FAILED(rc)) throw rc;
2728
2729 /* Set the VRAM */
2730 rc = pNewMachine->COMSETTER(VRAMSize)(vramVBox);
2731 if (FAILED(rc)) throw rc;
2732
2733 // I/O APIC: Generic OVF has no setting for this. Enable it if we
2734 // import a Windows VM because if if Windows was installed without IOAPIC,
2735 // it will not mind finding an one later on, but if Windows was installed
2736 // _with_ an IOAPIC, it will bluescreen if it's not found
2737 if (!stack.fForceIOAPIC)
2738 {
2739 Bstr bstrFamilyId;
2740 rc = osType->COMGETTER(FamilyId)(bstrFamilyId.asOutParam());
2741 if (FAILED(rc)) throw rc;
2742 if (bstrFamilyId == "Windows")
2743 stack.fForceIOAPIC = true;
2744 }
2745
2746 if (stack.fForceIOAPIC)
2747 {
2748 ComPtr<IBIOSSettings> pBIOSSettings;
2749 rc = pNewMachine->COMGETTER(BIOSSettings)(pBIOSSettings.asOutParam());
2750 if (FAILED(rc)) throw rc;
2751
2752 rc = pBIOSSettings->COMSETTER(IOAPICEnabled)(TRUE);
2753 if (FAILED(rc)) throw rc;
2754 }
2755
2756 if (!stack.strAudioAdapter.isEmpty())
2757 if (stack.strAudioAdapter.compare("null", Utf8Str::CaseInsensitive) != 0)
2758 {
2759 uint32_t audio = RTStrToUInt32(stack.strAudioAdapter.c_str()); // should be 0 for AC97
2760 ComPtr<IAudioAdapter> audioAdapter;
2761 rc = pNewMachine->COMGETTER(AudioAdapter)(audioAdapter.asOutParam());
2762 if (FAILED(rc)) throw rc;
2763 rc = audioAdapter->COMSETTER(Enabled)(true);
2764 if (FAILED(rc)) throw rc;
2765 rc = audioAdapter->COMSETTER(AudioController)(static_cast<AudioControllerType_T>(audio));
2766 if (FAILED(rc)) throw rc;
2767 }
2768
2769#ifdef VBOX_WITH_USB
2770 /* USB Controller */
2771 if (stack.fUSBEnabled)
2772 {
2773 ComPtr<IUSBController> usbController;
2774 rc = pNewMachine->AddUSBController(Bstr("OHCI").raw(), USBControllerType_OHCI, usbController.asOutParam());
2775 if (FAILED(rc)) throw rc;
2776 }
2777#endif /* VBOX_WITH_USB */
2778
2779 /* Change the network adapters */
2780 uint32_t maxNetworkAdapters = Global::getMaxNetworkAdapters(ChipsetType_PIIX3);
2781
2782 std::list<VirtualSystemDescriptionEntry*> vsdeNW = vsdescThis->i_findByType(VirtualSystemDescriptionType_NetworkAdapter);
2783 if (vsdeNW.empty())
2784 {
2785 /* No network adapters, so we have to disable our default one */
2786 ComPtr<INetworkAdapter> nwVBox;
2787 rc = pNewMachine->GetNetworkAdapter(0, nwVBox.asOutParam());
2788 if (FAILED(rc)) throw rc;
2789 rc = nwVBox->COMSETTER(Enabled)(false);
2790 if (FAILED(rc)) throw rc;
2791 }
2792 else if (vsdeNW.size() > maxNetworkAdapters)
2793 throw setError(VBOX_E_FILE_ERROR,
2794 tr("Too many network adapters: OVF requests %d network adapters, "
2795 "but VirtualBox only supports %d"),
2796 vsdeNW.size(), maxNetworkAdapters);
2797 else
2798 {
2799 list<VirtualSystemDescriptionEntry*>::const_iterator nwIt;
2800 size_t a = 0;
2801 for (nwIt = vsdeNW.begin();
2802 nwIt != vsdeNW.end();
2803 ++nwIt, ++a)
2804 {
2805 const VirtualSystemDescriptionEntry* pvsys = *nwIt;
2806
2807 const Utf8Str &nwTypeVBox = pvsys->strVBoxCurrent;
2808 uint32_t tt1 = RTStrToUInt32(nwTypeVBox.c_str());
2809 ComPtr<INetworkAdapter> pNetworkAdapter;
2810 rc = pNewMachine->GetNetworkAdapter((ULONG)a, pNetworkAdapter.asOutParam());
2811 if (FAILED(rc)) throw rc;
2812 /* Enable the network card & set the adapter type */
2813 rc = pNetworkAdapter->COMSETTER(Enabled)(true);
2814 if (FAILED(rc)) throw rc;
2815 rc = pNetworkAdapter->COMSETTER(AdapterType)(static_cast<NetworkAdapterType_T>(tt1));
2816 if (FAILED(rc)) throw rc;
2817
2818 // default is NAT; change to "bridged" if extra conf says so
2819 if (pvsys->strExtraConfigCurrent.endsWith("type=Bridged", Utf8Str::CaseInsensitive))
2820 {
2821 /* Attach to the right interface */
2822 rc = pNetworkAdapter->COMSETTER(AttachmentType)(NetworkAttachmentType_Bridged);
2823 if (FAILED(rc)) throw rc;
2824 ComPtr<IHost> host;
2825 rc = mVirtualBox->COMGETTER(Host)(host.asOutParam());
2826 if (FAILED(rc)) throw rc;
2827 com::SafeIfaceArray<IHostNetworkInterface> nwInterfaces;
2828 rc = host->COMGETTER(NetworkInterfaces)(ComSafeArrayAsOutParam(nwInterfaces));
2829 if (FAILED(rc)) throw rc;
2830 // We search for the first host network interface which
2831 // is usable for bridged networking
2832 for (size_t j = 0;
2833 j < nwInterfaces.size();
2834 ++j)
2835 {
2836 HostNetworkInterfaceType_T itype;
2837 rc = nwInterfaces[j]->COMGETTER(InterfaceType)(&itype);
2838 if (FAILED(rc)) throw rc;
2839 if (itype == HostNetworkInterfaceType_Bridged)
2840 {
2841 Bstr name;
2842 rc = nwInterfaces[j]->COMGETTER(Name)(name.asOutParam());
2843 if (FAILED(rc)) throw rc;
2844 /* Set the interface name to attach to */
2845 rc = pNetworkAdapter->COMSETTER(BridgedInterface)(name.raw());
2846 if (FAILED(rc)) throw rc;
2847 break;
2848 }
2849 }
2850 }
2851 /* Next test for host only interfaces */
2852 else if (pvsys->strExtraConfigCurrent.endsWith("type=HostOnly", Utf8Str::CaseInsensitive))
2853 {
2854 /* Attach to the right interface */
2855 rc = pNetworkAdapter->COMSETTER(AttachmentType)(NetworkAttachmentType_HostOnly);
2856 if (FAILED(rc)) throw rc;
2857 ComPtr<IHost> host;
2858 rc = mVirtualBox->COMGETTER(Host)(host.asOutParam());
2859 if (FAILED(rc)) throw rc;
2860 com::SafeIfaceArray<IHostNetworkInterface> nwInterfaces;
2861 rc = host->COMGETTER(NetworkInterfaces)(ComSafeArrayAsOutParam(nwInterfaces));
2862 if (FAILED(rc)) throw rc;
2863 // We search for the first host network interface which
2864 // is usable for host only networking
2865 for (size_t j = 0;
2866 j < nwInterfaces.size();
2867 ++j)
2868 {
2869 HostNetworkInterfaceType_T itype;
2870 rc = nwInterfaces[j]->COMGETTER(InterfaceType)(&itype);
2871 if (FAILED(rc)) throw rc;
2872 if (itype == HostNetworkInterfaceType_HostOnly)
2873 {
2874 Bstr name;
2875 rc = nwInterfaces[j]->COMGETTER(Name)(name.asOutParam());
2876 if (FAILED(rc)) throw rc;
2877 /* Set the interface name to attach to */
2878 rc = pNetworkAdapter->COMSETTER(HostOnlyInterface)(name.raw());
2879 if (FAILED(rc)) throw rc;
2880 break;
2881 }
2882 }
2883 }
2884 /* Next test for internal interfaces */
2885 else if (pvsys->strExtraConfigCurrent.endsWith("type=Internal", Utf8Str::CaseInsensitive))
2886 {
2887 /* Attach to the right interface */
2888 rc = pNetworkAdapter->COMSETTER(AttachmentType)(NetworkAttachmentType_Internal);
2889 if (FAILED(rc)) throw rc;
2890 }
2891 /* Next test for Generic interfaces */
2892 else if (pvsys->strExtraConfigCurrent.endsWith("type=Generic", Utf8Str::CaseInsensitive))
2893 {
2894 /* Attach to the right interface */
2895 rc = pNetworkAdapter->COMSETTER(AttachmentType)(NetworkAttachmentType_Generic);
2896 if (FAILED(rc)) throw rc;
2897 }
2898
2899 /* Next test for NAT network interfaces */
2900 else if (pvsys->strExtraConfigCurrent.endsWith("type=NATNetwork", Utf8Str::CaseInsensitive))
2901 {
2902 /* Attach to the right interface */
2903 rc = pNetworkAdapter->COMSETTER(AttachmentType)(NetworkAttachmentType_NATNetwork);
2904 if (FAILED(rc)) throw rc;
2905 com::SafeIfaceArray<INATNetwork> nwNATNetworks;
2906 rc = mVirtualBox->COMGETTER(NATNetworks)(ComSafeArrayAsOutParam(nwNATNetworks));
2907 if (FAILED(rc)) throw rc;
2908 // Pick the first NAT network (if there is any)
2909 if (nwNATNetworks.size())
2910 {
2911 Bstr name;
2912 rc = nwNATNetworks[0]->COMGETTER(NetworkName)(name.asOutParam());
2913 if (FAILED(rc)) throw rc;
2914 /* Set the NAT network name to attach to */
2915 rc = pNetworkAdapter->COMSETTER(NATNetwork)(name.raw());
2916 if (FAILED(rc)) throw rc;
2917 break;
2918 }
2919 }
2920 }
2921 }
2922
2923 // IDE Hard disk controller
2924 std::list<VirtualSystemDescriptionEntry*> vsdeHDCIDE =
2925 vsdescThis->i_findByType(VirtualSystemDescriptionType_HardDiskControllerIDE);
2926 /*
2927 * In OVF (at least VMware's version of it), an IDE controller has two ports,
2928 * so VirtualBox's single IDE controller with two channels and two ports each counts as
2929 * two OVF IDE controllers -- so we accept one or two such IDE controllers
2930 */
2931 size_t cIDEControllers = vsdeHDCIDE.size();
2932 if (cIDEControllers > 2)
2933 throw setError(VBOX_E_FILE_ERROR,
2934 tr("Too many IDE controllers in OVF; import facility only supports two"));
2935 if (!vsdeHDCIDE.empty())
2936 {
2937 // one or two IDE controllers present in OVF: add one VirtualBox controller
2938 ComPtr<IStorageController> pController;
2939 rc = pNewMachine->AddStorageController(Bstr("IDE Controller").raw(), StorageBus_IDE, pController.asOutParam());
2940 if (FAILED(rc)) throw rc;
2941
2942 const char *pcszIDEType = vsdeHDCIDE.front()->strVBoxCurrent.c_str();
2943 if (!strcmp(pcszIDEType, "PIIX3"))
2944 rc = pController->COMSETTER(ControllerType)(StorageControllerType_PIIX3);
2945 else if (!strcmp(pcszIDEType, "PIIX4"))
2946 rc = pController->COMSETTER(ControllerType)(StorageControllerType_PIIX4);
2947 else if (!strcmp(pcszIDEType, "ICH6"))
2948 rc = pController->COMSETTER(ControllerType)(StorageControllerType_ICH6);
2949 else
2950 throw setError(VBOX_E_FILE_ERROR,
2951 tr("Invalid IDE controller type \"%s\""),
2952 pcszIDEType);
2953 if (FAILED(rc)) throw rc;
2954 }
2955
2956 /* Hard disk controller SATA */
2957 std::list<VirtualSystemDescriptionEntry*> vsdeHDCSATA =
2958 vsdescThis->i_findByType(VirtualSystemDescriptionType_HardDiskControllerSATA);
2959 if (vsdeHDCSATA.size() > 1)
2960 throw setError(VBOX_E_FILE_ERROR,
2961 tr("Too many SATA controllers in OVF; import facility only supports one"));
2962 if (!vsdeHDCSATA.empty())
2963 {
2964 ComPtr<IStorageController> pController;
2965 const Utf8Str &hdcVBox = vsdeHDCSATA.front()->strVBoxCurrent;
2966 if (hdcVBox == "AHCI")
2967 {
2968 rc = pNewMachine->AddStorageController(Bstr("SATA Controller").raw(),
2969 StorageBus_SATA,
2970 pController.asOutParam());
2971 if (FAILED(rc)) throw rc;
2972 }
2973 else
2974 throw setError(VBOX_E_FILE_ERROR,
2975 tr("Invalid SATA controller type \"%s\""),
2976 hdcVBox.c_str());
2977 }
2978
2979 /* Hard disk controller SCSI */
2980 std::list<VirtualSystemDescriptionEntry*> vsdeHDCSCSI =
2981 vsdescThis->i_findByType(VirtualSystemDescriptionType_HardDiskControllerSCSI);
2982 if (vsdeHDCSCSI.size() > 1)
2983 throw setError(VBOX_E_FILE_ERROR,
2984 tr("Too many SCSI controllers in OVF; import facility only supports one"));
2985 if (!vsdeHDCSCSI.empty())
2986 {
2987 ComPtr<IStorageController> pController;
2988 Bstr bstrName(L"SCSI Controller");
2989 StorageBus_T busType = StorageBus_SCSI;
2990 StorageControllerType_T controllerType;
2991 const Utf8Str &hdcVBox = vsdeHDCSCSI.front()->strVBoxCurrent;
2992 if (hdcVBox == "LsiLogic")
2993 controllerType = StorageControllerType_LsiLogic;
2994 else if (hdcVBox == "LsiLogicSas")
2995 {
2996 // OVF treats LsiLogicSas as a SCSI controller but VBox considers it a class of its own
2997 bstrName = L"SAS Controller";
2998 busType = StorageBus_SAS;
2999 controllerType = StorageControllerType_LsiLogicSas;
3000 }
3001 else if (hdcVBox == "BusLogic")
3002 controllerType = StorageControllerType_BusLogic;
3003 else
3004 throw setError(VBOX_E_FILE_ERROR,
3005 tr("Invalid SCSI controller type \"%s\""),
3006 hdcVBox.c_str());
3007
3008 rc = pNewMachine->AddStorageController(bstrName.raw(), busType, pController.asOutParam());
3009 if (FAILED(rc)) throw rc;
3010 rc = pController->COMSETTER(ControllerType)(controllerType);
3011 if (FAILED(rc)) throw rc;
3012 }
3013
3014 /* Hard disk controller SAS */
3015 std::list<VirtualSystemDescriptionEntry*> vsdeHDCSAS =
3016 vsdescThis->i_findByType(VirtualSystemDescriptionType_HardDiskControllerSAS);
3017 if (vsdeHDCSAS.size() > 1)
3018 throw setError(VBOX_E_FILE_ERROR,
3019 tr("Too many SAS controllers in OVF; import facility only supports one"));
3020 if (!vsdeHDCSAS.empty())
3021 {
3022 ComPtr<IStorageController> pController;
3023 rc = pNewMachine->AddStorageController(Bstr(L"SAS Controller").raw(),
3024 StorageBus_SAS,
3025 pController.asOutParam());
3026 if (FAILED(rc)) throw rc;
3027 rc = pController->COMSETTER(ControllerType)(StorageControllerType_LsiLogicSas);
3028 if (FAILED(rc)) throw rc;
3029 }
3030
3031 /* Now its time to register the machine before we add any hard disks */
3032 rc = mVirtualBox->RegisterMachine(pNewMachine);
3033 if (FAILED(rc)) throw rc;
3034
3035 // store new machine for roll-back in case of errors
3036 Bstr bstrNewMachineId;
3037 rc = pNewMachine->COMGETTER(Id)(bstrNewMachineId.asOutParam());
3038 if (FAILED(rc)) throw rc;
3039 Guid uuidNewMachine(bstrNewMachineId);
3040 m->llGuidsMachinesCreated.push_back(uuidNewMachine);
3041
3042 // Add floppies and CD-ROMs to the appropriate controllers.
3043 std::list<VirtualSystemDescriptionEntry*> vsdeFloppy = vsdescThis->i_findByType(VirtualSystemDescriptionType_Floppy);
3044 if (vsdeFloppy.size() > 1)
3045 throw setError(VBOX_E_FILE_ERROR,
3046 tr("Too many floppy controllers in OVF; import facility only supports one"));
3047 std::list<VirtualSystemDescriptionEntry*> vsdeCDROM = vsdescThis->i_findByType(VirtualSystemDescriptionType_CDROM);
3048 if ( !vsdeFloppy.empty()
3049 || !vsdeCDROM.empty()
3050 )
3051 {
3052 // If there's an error here we need to close the session, so
3053 // we need another try/catch block.
3054
3055 try
3056 {
3057 // to attach things we need to open a session for the new machine
3058 rc = pNewMachine->LockMachine(stack.pSession, LockType_Write);
3059 if (FAILED(rc)) throw rc;
3060 stack.fSessionOpen = true;
3061
3062 ComPtr<IMachine> sMachine;
3063 rc = stack.pSession->COMGETTER(Machine)(sMachine.asOutParam());
3064 if (FAILED(rc)) throw rc;
3065
3066 // floppy first
3067 if (vsdeFloppy.size() == 1)
3068 {
3069 ComPtr<IStorageController> pController;
3070 rc = sMachine->AddStorageController(Bstr("Floppy Controller").raw(),
3071 StorageBus_Floppy,
3072 pController.asOutParam());
3073 if (FAILED(rc)) throw rc;
3074
3075 Bstr bstrName;
3076 rc = pController->COMGETTER(Name)(bstrName.asOutParam());
3077 if (FAILED(rc)) throw rc;
3078
3079 // this is for rollback later
3080 MyHardDiskAttachment mhda;
3081 mhda.pMachine = pNewMachine;
3082 mhda.controllerType = bstrName;
3083 mhda.lControllerPort = 0;
3084 mhda.lDevice = 0;
3085
3086 Log(("Attaching floppy\n"));
3087
3088 rc = sMachine->AttachDevice(mhda.controllerType.raw(),
3089 mhda.lControllerPort,
3090 mhda.lDevice,
3091 DeviceType_Floppy,
3092 NULL);
3093 if (FAILED(rc)) throw rc;
3094
3095 stack.llHardDiskAttachments.push_back(mhda);
3096 }
3097
3098 rc = sMachine->SaveSettings();
3099 if (FAILED(rc)) throw rc;
3100
3101 // only now that we're done with all disks, close the session
3102 rc = stack.pSession->UnlockMachine();
3103 if (FAILED(rc)) throw rc;
3104 stack.fSessionOpen = false;
3105 }
3106 catch(HRESULT aRC)
3107 {
3108 com::ErrorInfo info;
3109
3110 if (stack.fSessionOpen)
3111 stack.pSession->UnlockMachine();
3112
3113 if (info.isFullAvailable())
3114 throw setError(aRC, Utf8Str(info.getText()).c_str());
3115 else
3116 throw setError(aRC, "Unknown error during OVF import");
3117 }
3118 }
3119
3120 // create the hard disks & connect them to the appropriate controllers
3121 std::list<VirtualSystemDescriptionEntry*> avsdeHDs = vsdescThis->i_findByType(VirtualSystemDescriptionType_HardDiskImage);
3122 if (!avsdeHDs.empty())
3123 {
3124 // If there's an error here we need to close the session, so
3125 // we need another try/catch block.
3126 try
3127 {
3128#ifdef LOG_ENABLED
3129 if (LogIsEnabled())
3130 {
3131 size_t i = 0;
3132 for (list<VirtualSystemDescriptionEntry*>::const_iterator itHD = avsdeHDs.begin();
3133 itHD != avsdeHDs.end(); ++itHD, i++)
3134 Log(("avsdeHDs[%zu]: strRef=%s strOvf=%s\n", i, (*itHD)->strRef.c_str(), (*itHD)->strOvf.c_str()));
3135 i = 0;
3136 for (ovf::DiskImagesMap::const_iterator itDisk = stack.mapDisks.begin(); itDisk != stack.mapDisks.end(); ++itDisk)
3137 Log(("mapDisks[%zu]: strDiskId=%s strHref=%s\n",
3138 i, itDisk->second.strDiskId.c_str(), itDisk->second.strHref.c_str()));
3139
3140 }
3141#endif
3142
3143 // to attach things we need to open a session for the new machine
3144 rc = pNewMachine->LockMachine(stack.pSession, LockType_Write);
3145 if (FAILED(rc)) throw rc;
3146 stack.fSessionOpen = true;
3147
3148 /* get VM name from virtual system description. Only one record is possible (size of list is equal 1). */
3149 std::list<VirtualSystemDescriptionEntry*> vmName = vsdescThis->i_findByType(VirtualSystemDescriptionType_Name);
3150 std::list<VirtualSystemDescriptionEntry*>::iterator vmNameIt = vmName.begin();
3151 VirtualSystemDescriptionEntry* vmNameEntry = *vmNameIt;
3152
3153
3154 ovf::DiskImagesMap::const_iterator oit = stack.mapDisks.begin();
3155 std::set<RTCString> disksResolvedNames;
3156
3157 uint32_t cImportedDisks = 0;
3158
3159 while (oit != stack.mapDisks.end() && cImportedDisks != avsdeHDs.size())
3160 {
3161/** @todo r=bird: Most of the code here is duplicated in the other machine
3162 * import method, factor out. */
3163 ovf::DiskImage diCurrent = oit->second;
3164
3165 Log(("diCurrent.strDiskId=%s diCurrent.strHref=%s\n", diCurrent.strDiskId.c_str(), diCurrent.strHref.c_str()));
3166 /* Iterate over all given disk images of the virtual system
3167 * disks description. We need to find the target disk path,
3168 * which could be changed by the user. */
3169 VirtualSystemDescriptionEntry *vsdeTargetHD = NULL;
3170 for (list<VirtualSystemDescriptionEntry*>::const_iterator itHD = avsdeHDs.begin();
3171 itHD != avsdeHDs.end();
3172 ++itHD)
3173 {
3174 VirtualSystemDescriptionEntry *vsdeHD = *itHD;
3175 if (vsdeHD->strRef == diCurrent.strDiskId)
3176 {
3177 vsdeTargetHD = vsdeHD;
3178 break;
3179 }
3180 }
3181 if (!vsdeTargetHD)
3182 {
3183 /* possible case if a disk image belongs to other virtual system (OVF package with multiple VMs inside) */
3184 Log1Warning(("OVA/OVF import: Disk image %s was missed during import of VM %s\n",
3185 oit->first.c_str(), vmNameEntry->strOvf.c_str()));
3186 NOREF(vmNameEntry);
3187 ++oit;
3188 continue;
3189 }
3190
3191 //diCurrent.strDiskId contains the disk identifier (e.g. "vmdisk1"), which should exist
3192 //in the virtual system's disks map under that ID and also in the global images map
3193 ovf::VirtualDisksMap::const_iterator itVDisk = vsysThis.mapVirtualDisks.find(diCurrent.strDiskId);
3194 if (itVDisk == vsysThis.mapVirtualDisks.end())
3195 throw setError(E_FAIL,
3196 tr("Internal inconsistency looking up disk image '%s'"),
3197 diCurrent.strHref.c_str());
3198
3199 /*
3200 * preliminary check availability of the image
3201 * This step is useful if image is placed in the OVA (TAR) package
3202 */
3203 if (stack.hVfsFssOva != NIL_RTVFSFSSTREAM)
3204 {
3205 /* It means that we possibly have imported the storage earlier on the previous loop steps*/
3206 std::set<RTCString>::const_iterator h = disksResolvedNames.find(diCurrent.strHref);
3207 if (h != disksResolvedNames.end())
3208 {
3209 /* Yes, disk name was found, we can skip it*/
3210 ++oit;
3211 continue;
3212 }
3213l_skipped:
3214 rc = i_preCheckImageAvailability(stack);
3215 if (SUCCEEDED(rc))
3216 {
3217 /* current opened file isn't the same as passed one */
3218 if (RTStrICmp(diCurrent.strHref.c_str(), stack.pszOvaLookAheadName) != 0)
3219 {
3220 /* availableImage contains the disk file reference (e.g. "disk1.vmdk"), which should
3221 * exist in the global images map.
3222 * And find the disk from the OVF's disk list */
3223 ovf::DiskImagesMap::const_iterator itDiskImage;
3224 for (itDiskImage = stack.mapDisks.begin();
3225 itDiskImage != stack.mapDisks.end();
3226 itDiskImage++)
3227 if (itDiskImage->second.strHref.compare(stack.pszOvaLookAheadName,
3228 Utf8Str::CaseInsensitive) == 0)
3229 break;
3230 if (itDiskImage == stack.mapDisks.end())
3231 {
3232 LogFunc(("Skipping '%s'\n", stack.pszOvaLookAheadName));
3233 RTVfsIoStrmRelease(stack.claimOvaLookAHead());
3234 goto l_skipped;
3235 }
3236
3237 /* replace with a new found disk image */
3238 diCurrent = *(&itDiskImage->second);
3239
3240 /*
3241 * Again iterate over all given disk images of the virtual system
3242 * disks description using the found disk image
3243 */
3244 for (list<VirtualSystemDescriptionEntry*>::const_iterator itHD = avsdeHDs.begin();
3245 itHD != avsdeHDs.end();
3246 ++itHD)
3247 {
3248 VirtualSystemDescriptionEntry *vsdeHD = *itHD;
3249 if (vsdeHD->strRef == diCurrent.strDiskId)
3250 {
3251 vsdeTargetHD = vsdeHD;
3252 break;
3253 }
3254 }
3255
3256 /*
3257 * in this case it's an error because something is wrong with the OVF description file.
3258 * May be VBox imports OVA package with wrong file sequence inside the archive.
3259 */
3260 if (!vsdeTargetHD)
3261 throw setError(E_FAIL,
3262 tr("Internal inconsistency looking up disk image '%s'"),
3263 diCurrent.strHref.c_str());
3264
3265 itVDisk = vsysThis.mapVirtualDisks.find(diCurrent.strDiskId);
3266 if (itVDisk == vsysThis.mapVirtualDisks.end())
3267 throw setError(E_FAIL,
3268 tr("Internal inconsistency looking up disk image '%s'"),
3269 diCurrent.strHref.c_str());
3270 }
3271 else
3272 {
3273 ++oit;
3274 }
3275 }
3276 else
3277 {
3278 ++oit;
3279 continue;
3280 }
3281 }
3282 else
3283 {
3284 /* just continue with normal files*/
3285 ++oit;
3286 }
3287
3288 /* very important to store disk name for the next checks */
3289 disksResolvedNames.insert(diCurrent.strHref);
3290////// end of duplicated code.
3291 const ovf::VirtualDisk &ovfVdisk = itVDisk->second;
3292
3293 ComObjPtr<Medium> pTargetHD;
3294
3295 Utf8Str savedVBoxCurrent = vsdeTargetHD->strVBoxCurrent;
3296
3297 i_importOneDiskImage(diCurrent,
3298 &vsdeTargetHD->strVBoxCurrent,
3299 pTargetHD,
3300 stack);
3301
3302 // now use the new uuid to attach the disk image to our new machine
3303 ComPtr<IMachine> sMachine;
3304 rc = stack.pSession->COMGETTER(Machine)(sMachine.asOutParam());
3305 if (FAILED(rc))
3306 throw rc;
3307
3308 // find the hard disk controller to which we should attach
3309 ovf::HardDiskController hdc = (*vsysThis.mapControllers.find(ovfVdisk.idController)).second;
3310
3311 // this is for rollback later
3312 MyHardDiskAttachment mhda;
3313 mhda.pMachine = pNewMachine;
3314
3315 i_convertDiskAttachmentValues(hdc,
3316 ovfVdisk.ulAddressOnParent,
3317 mhda.controllerType, // Bstr
3318 mhda.lControllerPort,
3319 mhda.lDevice);
3320
3321 Log(("Attaching disk %s to port %d on device %d\n",
3322 vsdeTargetHD->strVBoxCurrent.c_str(), mhda.lControllerPort, mhda.lDevice));
3323
3324 ComObjPtr<MediumFormat> mediumFormat;
3325 rc = i_findMediumFormatFromDiskImage(diCurrent, mediumFormat);
3326 if (FAILED(rc))
3327 throw rc;
3328
3329 Bstr bstrFormatName;
3330 rc = mediumFormat->COMGETTER(Name)(bstrFormatName.asOutParam());
3331 if (FAILED(rc))
3332 throw rc;
3333
3334 Utf8Str vdf = Utf8Str(bstrFormatName);
3335
3336 if (vdf.compare("RAW", Utf8Str::CaseInsensitive) == 0)
3337 {
3338 ComPtr<IMedium> dvdImage(pTargetHD);
3339
3340 rc = mVirtualBox->OpenMedium(Bstr(vsdeTargetHD->strVBoxCurrent).raw(),
3341 DeviceType_DVD,
3342 AccessMode_ReadWrite,
3343 false,
3344 dvdImage.asOutParam());
3345
3346 if (FAILED(rc))
3347 throw rc;
3348
3349 rc = sMachine->AttachDevice(mhda.controllerType.raw(),// wstring name
3350 mhda.lControllerPort, // long controllerPort
3351 mhda.lDevice, // long device
3352 DeviceType_DVD, // DeviceType_T type
3353 dvdImage);
3354 if (FAILED(rc))
3355 throw rc;
3356 }
3357 else
3358 {
3359 rc = sMachine->AttachDevice(mhda.controllerType.raw(),// wstring name
3360 mhda.lControllerPort, // long controllerPort
3361 mhda.lDevice, // long device
3362 DeviceType_HardDisk, // DeviceType_T type
3363 pTargetHD);
3364
3365 if (FAILED(rc))
3366 throw rc;
3367 }
3368
3369 stack.llHardDiskAttachments.push_back(mhda);
3370
3371 rc = sMachine->SaveSettings();
3372 if (FAILED(rc))
3373 throw rc;
3374
3375 /* restore */
3376 vsdeTargetHD->strVBoxCurrent = savedVBoxCurrent;
3377
3378 ++cImportedDisks;
3379
3380 } // end while(oit != stack.mapDisks.end())
3381
3382 /*
3383 * quantity of the imported disks isn't equal to the size of the avsdeHDs list.
3384 */
3385 if(cImportedDisks < avsdeHDs.size())
3386 {
3387 Log1Warning(("Not all disk images were imported for VM %s. Check OVF description file.",
3388 vmNameEntry->strOvf.c_str()));
3389 }
3390
3391 // only now that we're done with all disks, close the session
3392 rc = stack.pSession->UnlockMachine();
3393 if (FAILED(rc))
3394 throw rc;
3395 stack.fSessionOpen = false;
3396 }
3397 catch(HRESULT aRC)
3398 {
3399 com::ErrorInfo info;
3400 if (stack.fSessionOpen)
3401 stack.pSession->UnlockMachine();
3402
3403 if (info.isFullAvailable())
3404 throw setError(aRC, Utf8Str(info.getText()).c_str());
3405 else
3406 throw setError(aRC, "Unknown error during OVF import");
3407 }
3408 }
3409 LogFlowFuncLeave();
3410}
3411
3412/**
3413 * Imports one OVF virtual system (described by a vbox:Machine tag represented by the given config
3414 * structure) into VirtualBox by creating an IMachine instance, which is returned.
3415 *
3416 * This throws HRESULT error codes for anything that goes wrong, in which case the caller must clean
3417 * up any leftovers from this function. For this, the given ImportStack instance has received information
3418 * about what needs cleaning up (to support rollback).
3419 *
3420 * The machine config stored in the settings::MachineConfigFile structure contains the UUIDs of
3421 * the disk attachments used by the machine when it was exported. We also add vbox:uuid attributes
3422 * to the OVF disks sections so we can look them up. While importing these UUIDs into a second host
3423 * will most probably work, reimporting them into the same host will cause conflicts, so we always
3424 * generate new ones on import. This involves the following:
3425 *
3426 * 1) Scan the machine config for disk attachments.
3427 *
3428 * 2) For each disk attachment found, look up the OVF disk image from the disk references section
3429 * and import the disk into VirtualBox, which creates a new UUID for it. In the machine config,
3430 * replace the old UUID with the new one.
3431 *
3432 * 3) Change the machine config according to the OVF virtual system descriptions, in case the
3433 * caller has modified them using setFinalValues().
3434 *
3435 * 4) Create the VirtualBox machine with the modfified machine config.
3436 *
3437 * @param config
3438 * @param pNewMachine
3439 * @param stack
3440 */
3441void Appliance::i_importVBoxMachine(ComObjPtr<VirtualSystemDescription> &vsdescThis,
3442 ComPtr<IMachine> &pReturnNewMachine,
3443 ImportStack &stack)
3444{
3445 LogFlowFuncEnter();
3446 Assert(vsdescThis->m->pConfig);
3447
3448 HRESULT rc = S_OK;
3449
3450 settings::MachineConfigFile &config = *vsdescThis->m->pConfig;
3451
3452 /*
3453 * step 1): modify machine config according to OVF config, in case the user
3454 * has modified them using setFinalValues()
3455 */
3456
3457 /* OS Type */
3458 config.machineUserData.strOsType = stack.strOsTypeVBox;
3459 /* Description */
3460 config.machineUserData.strDescription = stack.strDescription;
3461 /* CPU count & extented attributes */
3462 config.hardwareMachine.cCPUs = stack.cCPUs;
3463 if (stack.fForceIOAPIC)
3464 config.hardwareMachine.fHardwareVirt = true;
3465 if (stack.fForceIOAPIC)
3466 config.hardwareMachine.biosSettings.fIOAPICEnabled = true;
3467 /* RAM size */
3468 config.hardwareMachine.ulMemorySizeMB = stack.ulMemorySizeMB;
3469
3470/*
3471 <const name="HardDiskControllerIDE" value="14" />
3472 <const name="HardDiskControllerSATA" value="15" />
3473 <const name="HardDiskControllerSCSI" value="16" />
3474 <const name="HardDiskControllerSAS" value="17" />
3475*/
3476
3477#ifdef VBOX_WITH_USB
3478 /* USB controller */
3479 if (stack.fUSBEnabled)
3480 {
3481 /** @todo r=klaus add support for arbitrary USB controller types, this can't handle
3482 * multiple controllers due to its design anyway */
3483 /* usually the OHCI controller is enabled already, need to check */
3484 bool fOHCIEnabled = false;
3485 settings::USBControllerList &llUSBControllers = config.hardwareMachine.usbSettings.llUSBControllers;
3486 settings::USBControllerList::iterator it;
3487 for (it = llUSBControllers.begin(); it != llUSBControllers.end(); ++it)
3488 {
3489 if (it->enmType == USBControllerType_OHCI)
3490 {
3491 fOHCIEnabled = true;
3492 break;
3493 }
3494 }
3495
3496 if (!fOHCIEnabled)
3497 {
3498 settings::USBController ctrl;
3499 ctrl.strName = "OHCI";
3500 ctrl.enmType = USBControllerType_OHCI;
3501
3502 llUSBControllers.push_back(ctrl);
3503 }
3504 }
3505 else
3506 config.hardwareMachine.usbSettings.llUSBControllers.clear();
3507#endif
3508 /* Audio adapter */
3509 if (stack.strAudioAdapter.isNotEmpty())
3510 {
3511 config.hardwareMachine.audioAdapter.fEnabled = true;
3512 config.hardwareMachine.audioAdapter.controllerType = (AudioControllerType_T)stack.strAudioAdapter.toUInt32();
3513 }
3514 else
3515 config.hardwareMachine.audioAdapter.fEnabled = false;
3516 /* Network adapter */
3517 settings::NetworkAdaptersList &llNetworkAdapters = config.hardwareMachine.llNetworkAdapters;
3518 /* First disable all network cards, they will be enabled below again. */
3519 settings::NetworkAdaptersList::iterator it1;
3520 bool fKeepAllMACs = m->optListImport.contains(ImportOptions_KeepAllMACs);
3521 bool fKeepNATMACs = m->optListImport.contains(ImportOptions_KeepNATMACs);
3522 for (it1 = llNetworkAdapters.begin(); it1 != llNetworkAdapters.end(); ++it1)
3523 {
3524 it1->fEnabled = false;
3525 if (!( fKeepAllMACs
3526 || (fKeepNATMACs && it1->mode == NetworkAttachmentType_NAT)
3527 || (fKeepNATMACs && it1->mode == NetworkAttachmentType_NATNetwork)))
3528 Host::i_generateMACAddress(it1->strMACAddress);
3529 }
3530 /* Now iterate over all network entries. */
3531 std::list<VirtualSystemDescriptionEntry*> avsdeNWs = vsdescThis->i_findByType(VirtualSystemDescriptionType_NetworkAdapter);
3532 if (!avsdeNWs.empty())
3533 {
3534 /* Iterate through all network adapter entries and search for the
3535 * corresponding one in the machine config. If one is found, configure
3536 * it based on the user settings. */
3537 list<VirtualSystemDescriptionEntry*>::const_iterator itNW;
3538 for (itNW = avsdeNWs.begin();
3539 itNW != avsdeNWs.end();
3540 ++itNW)
3541 {
3542 VirtualSystemDescriptionEntry *vsdeNW = *itNW;
3543 if ( vsdeNW->strExtraConfigCurrent.startsWith("slot=", Utf8Str::CaseInsensitive)
3544 && vsdeNW->strExtraConfigCurrent.length() > 6)
3545 {
3546 uint32_t iSlot = vsdeNW->strExtraConfigCurrent.substr(5, 1).toUInt32();
3547 /* Iterate through all network adapters in the machine config. */
3548 for (it1 = llNetworkAdapters.begin();
3549 it1 != llNetworkAdapters.end();
3550 ++it1)
3551 {
3552 /* Compare the slots. */
3553 if (it1->ulSlot == iSlot)
3554 {
3555 it1->fEnabled = true;
3556 it1->type = (NetworkAdapterType_T)vsdeNW->strVBoxCurrent.toUInt32();
3557 break;
3558 }
3559 }
3560 }
3561 }
3562 }
3563
3564 /* Floppy controller */
3565 bool fFloppy = vsdescThis->i_findByType(VirtualSystemDescriptionType_Floppy).size() > 0;
3566 /* DVD controller */
3567 bool fDVD = vsdescThis->i_findByType(VirtualSystemDescriptionType_CDROM).size() > 0;
3568 /* Iterate over all storage controller check the attachments and remove
3569 * them when necessary. Also detect broken configs with more than one
3570 * attachment. Old VirtualBox versions (prior to 3.2.10) had all disk
3571 * attachments pointing to the last hard disk image, which causes import
3572 * failures. A long fixed bug, however the OVF files are long lived. */
3573 settings::StorageControllersList &llControllers = config.storageMachine.llStorageControllers;
3574 Guid hdUuid;
3575 uint32_t cDisks = 0;
3576 bool fInconsistent = false;
3577 bool fRepairDuplicate = false;
3578 settings::StorageControllersList::iterator it3;
3579 for (it3 = llControllers.begin();
3580 it3 != llControllers.end();
3581 ++it3)
3582 {
3583 settings::AttachedDevicesList &llAttachments = it3->llAttachedDevices;
3584 settings::AttachedDevicesList::iterator it4 = llAttachments.begin();
3585 while (it4 != llAttachments.end())
3586 {
3587 if ( ( !fDVD
3588 && it4->deviceType == DeviceType_DVD)
3589 ||
3590 ( !fFloppy
3591 && it4->deviceType == DeviceType_Floppy))
3592 {
3593 it4 = llAttachments.erase(it4);
3594 continue;
3595 }
3596 else if (it4->deviceType == DeviceType_HardDisk)
3597 {
3598 const Guid &thisUuid = it4->uuid;
3599 cDisks++;
3600 if (cDisks == 1)
3601 {
3602 if (hdUuid.isZero())
3603 hdUuid = thisUuid;
3604 else
3605 fInconsistent = true;
3606 }
3607 else
3608 {
3609 if (thisUuid.isZero())
3610 fInconsistent = true;
3611 else if (thisUuid == hdUuid)
3612 fRepairDuplicate = true;
3613 }
3614 }
3615 ++it4;
3616 }
3617 }
3618 /* paranoia... */
3619 if (fInconsistent || cDisks == 1)
3620 fRepairDuplicate = false;
3621
3622 /*
3623 * step 2: scan the machine config for media attachments
3624 */
3625 /* get VM name from virtual system description. Only one record is possible (size of list is equal 1). */
3626 std::list<VirtualSystemDescriptionEntry*> vmName = vsdescThis->i_findByType(VirtualSystemDescriptionType_Name);
3627 std::list<VirtualSystemDescriptionEntry*>::iterator vmNameIt = vmName.begin();
3628 VirtualSystemDescriptionEntry* vmNameEntry = *vmNameIt;
3629
3630 /* Get all hard disk descriptions. */
3631 std::list<VirtualSystemDescriptionEntry*> avsdeHDs = vsdescThis->i_findByType(VirtualSystemDescriptionType_HardDiskImage);
3632 std::list<VirtualSystemDescriptionEntry*>::iterator avsdeHDsIt = avsdeHDs.begin();
3633 /* paranoia - if there is no 1:1 match do not try to repair. */
3634 if (cDisks != avsdeHDs.size())
3635 fRepairDuplicate = false;
3636
3637 // there must be an image in the OVF disk structs with the same UUID
3638
3639 ovf::DiskImagesMap::const_iterator oit = stack.mapDisks.begin();
3640 std::set<RTCString> disksResolvedNames;
3641
3642 uint32_t cImportedDisks = 0;
3643
3644 while (oit != stack.mapDisks.end() && cImportedDisks != avsdeHDs.size())
3645 {
3646/** @todo r=bird: Most of the code here is duplicated in the other machine
3647 * import method, factor out. */
3648 ovf::DiskImage diCurrent = oit->second;
3649
3650 Log(("diCurrent.strDiskId=%s diCurrent.strHref=%s\n", diCurrent.strDiskId.c_str(), diCurrent.strHref.c_str()));
3651
3652 /* Iterate over all given disk images of the virtual system
3653 * disks description. We need to find the target disk path,
3654 * which could be changed by the user. */
3655 VirtualSystemDescriptionEntry *vsdeTargetHD = NULL;
3656 for (list<VirtualSystemDescriptionEntry*>::const_iterator itHD = avsdeHDs.begin();
3657 itHD != avsdeHDs.end();
3658 ++itHD)
3659 {
3660 VirtualSystemDescriptionEntry *vsdeHD = *itHD;
3661 if (vsdeHD->strRef == oit->first)
3662 {
3663 vsdeTargetHD = vsdeHD;
3664 break;
3665 }
3666 }
3667 if (!vsdeTargetHD)
3668 {
3669 /* possible case if a disk image belongs to other virtual system (OVF package with multiple VMs inside) */
3670 Log1Warning(("OVA/OVF import: Disk image %s was missed during import of VM %s\n",
3671 oit->first.c_str(), vmNameEntry->strOvf.c_str()));
3672 NOREF(vmNameEntry);
3673 ++oit;
3674 continue;
3675 }
3676
3677
3678
3679
3680
3681
3682
3683
3684
3685 /*
3686 * preliminary check availability of the image
3687 * This step is useful if image is placed in the OVA (TAR) package
3688 */
3689 if (stack.hVfsFssOva != NIL_RTVFSFSSTREAM)
3690 {
3691 /* It means that we possibly have imported the storage earlier on a previous loop step. */
3692 std::set<RTCString>::const_iterator h = disksResolvedNames.find(diCurrent.strHref);
3693 if (h != disksResolvedNames.end())
3694 {
3695 /* Yes, disk name was found, we can skip it*/
3696 ++oit;
3697 continue;
3698 }
3699l_skipped:
3700 rc = i_preCheckImageAvailability(stack);
3701 if (SUCCEEDED(rc))
3702 {
3703 /* current opened file isn't the same as passed one */
3704 if (RTStrICmp(diCurrent.strHref.c_str(), stack.pszOvaLookAheadName) != 0)
3705 {
3706 // availableImage contains the disk identifier (e.g. "vmdisk1"), which should exist
3707 // in the virtual system's disks map under that ID and also in the global images map
3708 // and find the disk from the OVF's disk list
3709 ovf::DiskImagesMap::const_iterator itDiskImage;
3710 for (itDiskImage = stack.mapDisks.begin();
3711 itDiskImage != stack.mapDisks.end();
3712 itDiskImage++)
3713 if (itDiskImage->second.strHref.compare(stack.pszOvaLookAheadName,
3714 Utf8Str::CaseInsensitive) == 0)
3715 break;
3716 if (itDiskImage == stack.mapDisks.end())
3717 {
3718 LogFunc(("Skipping '%s'\n", stack.pszOvaLookAheadName));
3719 RTVfsIoStrmRelease(stack.claimOvaLookAHead());
3720 goto l_skipped;
3721 }
3722 //throw setError(E_FAIL,
3723 // tr("Internal inconsistency looking up disk image '%s'. "
3724 // "Check compliance OVA package structure and file names "
3725 // "references in the section <References> in the OVF file."),
3726 // stack.pszOvaLookAheadName);
3727
3728 /* replace with a new found disk image */
3729 diCurrent = *(&itDiskImage->second);
3730
3731 /*
3732 * Again iterate over all given disk images of the virtual system
3733 * disks description using the found disk image
3734 */
3735 for (list<VirtualSystemDescriptionEntry*>::const_iterator itHD = avsdeHDs.begin();
3736 itHD != avsdeHDs.end();
3737 ++itHD)
3738 {
3739 VirtualSystemDescriptionEntry *vsdeHD = *itHD;
3740 if (vsdeHD->strRef == diCurrent.strDiskId)
3741 {
3742 vsdeTargetHD = vsdeHD;
3743 break;
3744 }
3745 }
3746
3747 /*
3748 * in this case it's an error because something is wrong with the OVF description file.
3749 * May be VBox imports OVA package with wrong file sequence inside the archive.
3750 */
3751 if (!vsdeTargetHD)
3752 throw setError(E_FAIL,
3753 tr("Internal inconsistency looking up disk image '%s'"),
3754 diCurrent.strHref.c_str());
3755
3756
3757
3758
3759
3760 }
3761 else
3762 {
3763 ++oit;
3764 }
3765 }
3766 else
3767 {
3768 ++oit;
3769 continue;
3770 }
3771 }
3772 else
3773 {
3774 /* just continue with normal files*/
3775 ++oit;
3776 }
3777
3778 /* Important! to store disk name for the next checks */
3779 disksResolvedNames.insert(diCurrent.strHref);
3780////// end of duplicated code.
3781 // there must be an image in the OVF disk structs with the same UUID
3782 bool fFound = false;
3783 Utf8Str strUuid;
3784
3785 // for each storage controller...
3786 for (settings::StorageControllersList::iterator sit = config.storageMachine.llStorageControllers.begin();
3787 sit != config.storageMachine.llStorageControllers.end();
3788 ++sit)
3789 {
3790 settings::StorageController &sc = *sit;
3791
3792 // find the OVF virtual system description entry for this storage controller
3793 switch (sc.storageBus)
3794 {
3795 case StorageBus_SATA:
3796 break;
3797 case StorageBus_SCSI:
3798 break;
3799 case StorageBus_IDE:
3800 break;
3801 case StorageBus_SAS:
3802 break;
3803 }
3804
3805 // for each medium attachment to this controller...
3806 for (settings::AttachedDevicesList::iterator dit = sc.llAttachedDevices.begin();
3807 dit != sc.llAttachedDevices.end();
3808 ++dit)
3809 {
3810 settings::AttachedDevice &d = *dit;
3811
3812 if (d.uuid.isZero())
3813 // empty DVD and floppy media
3814 continue;
3815
3816 // When repairing a broken VirtualBox xml config section (written
3817 // by VirtualBox versions earlier than 3.2.10) assume the disks
3818 // show up in the same order as in the OVF description.
3819 if (fRepairDuplicate)
3820 {
3821 VirtualSystemDescriptionEntry *vsdeHD = *avsdeHDsIt;
3822 ovf::DiskImagesMap::const_iterator itDiskImage = stack.mapDisks.find(vsdeHD->strRef);
3823 if (itDiskImage != stack.mapDisks.end())
3824 {
3825 const ovf::DiskImage &di = itDiskImage->second;
3826 d.uuid = Guid(di.uuidVBox);
3827 }
3828 ++avsdeHDsIt;
3829 }
3830
3831 // convert the Guid to string
3832 strUuid = d.uuid.toString();
3833
3834 if (diCurrent.uuidVBox != strUuid)
3835 {
3836 continue;
3837 }
3838
3839 /*
3840 * step 3: import disk
3841 */
3842 Utf8Str savedVBoxCurrent = vsdeTargetHD->strVBoxCurrent;
3843 ComObjPtr<Medium> pTargetHD;
3844
3845 i_importOneDiskImage(diCurrent,
3846 &vsdeTargetHD->strVBoxCurrent,
3847 pTargetHD,
3848 stack);
3849
3850 Bstr hdId;
3851
3852 ComObjPtr<MediumFormat> mediumFormat;
3853 rc = i_findMediumFormatFromDiskImage(diCurrent, mediumFormat);
3854 if (FAILED(rc))
3855 throw rc;
3856
3857 Bstr bstrFormatName;
3858 rc = mediumFormat->COMGETTER(Name)(bstrFormatName.asOutParam());
3859 if (FAILED(rc))
3860 throw rc;
3861
3862 Utf8Str vdf = Utf8Str(bstrFormatName);
3863
3864 if (vdf.compare("RAW", Utf8Str::CaseInsensitive) == 0)
3865 {
3866 ComPtr<IMedium> dvdImage(pTargetHD);
3867
3868 rc = mVirtualBox->OpenMedium(Bstr(vsdeTargetHD->strVBoxCurrent).raw(),
3869 DeviceType_DVD,
3870 AccessMode_ReadWrite,
3871 false,
3872 dvdImage.asOutParam());
3873
3874 if (FAILED(rc)) throw rc;
3875
3876 // ... and replace the old UUID in the machine config with the one of
3877 // the imported disk that was just created
3878 rc = dvdImage->COMGETTER(Id)(hdId.asOutParam());
3879 if (FAILED(rc)) throw rc;
3880 }
3881 else
3882 {
3883 // ... and replace the old UUID in the machine config with the one of
3884 // the imported disk that was just created
3885 rc = pTargetHD->COMGETTER(Id)(hdId.asOutParam());
3886 if (FAILED(rc)) throw rc;
3887 }
3888
3889 /* restore */
3890 vsdeTargetHD->strVBoxCurrent = savedVBoxCurrent;
3891
3892 /*
3893 * 1. saving original UUID for restoring in case of failure.
3894 * 2. replacement of original UUID by new UUID in the current VM config (settings::MachineConfigFile).
3895 */
3896 {
3897 rc = stack.saveOriginalUUIDOfAttachedDevice(d, Utf8Str(hdId));
3898 d.uuid = hdId;
3899 }
3900
3901 fFound = true;
3902 break;
3903 } // for (settings::AttachedDevicesList::const_iterator dit = sc.llAttachedDevices.begin();
3904 } // for (settings::StorageControllersList::const_iterator sit = config.storageMachine.llStorageControllers.begin();
3905
3906 // no disk with such a UUID found:
3907 if (!fFound)
3908 throw setError(E_FAIL,
3909 tr("<vbox:Machine> element in OVF contains a medium attachment for the disk image %s "
3910 "but the OVF describes no such image"),
3911 strUuid.c_str());
3912
3913 ++cImportedDisks;
3914
3915 }// while(oit != stack.mapDisks.end())
3916
3917
3918 /*
3919 * quantity of the imported disks isn't equal to the size of the avsdeHDs list.
3920 */
3921 if(cImportedDisks < avsdeHDs.size())
3922 {
3923 Log1Warning(("Not all disk images were imported for VM %s. Check OVF description file.",
3924 vmNameEntry->strOvf.c_str()));
3925 }
3926
3927 /*
3928 * step 4): create the machine and have it import the config
3929 */
3930
3931 ComObjPtr<Machine> pNewMachine;
3932 rc = pNewMachine.createObject();
3933 if (FAILED(rc)) throw rc;
3934
3935 // this magic constructor fills the new machine object with the MachineConfig
3936 // instance that we created from the vbox:Machine
3937 rc = pNewMachine->init(mVirtualBox,
3938 stack.strNameVBox,// name from OVF preparations; can be suffixed to avoid duplicates
3939 config); // the whole machine config
3940 if (FAILED(rc)) throw rc;
3941
3942 pReturnNewMachine = ComPtr<IMachine>(pNewMachine);
3943
3944 // and register it
3945 rc = mVirtualBox->RegisterMachine(pNewMachine);
3946 if (FAILED(rc)) throw rc;
3947
3948 // store new machine for roll-back in case of errors
3949 Bstr bstrNewMachineId;
3950 rc = pNewMachine->COMGETTER(Id)(bstrNewMachineId.asOutParam());
3951 if (FAILED(rc)) throw rc;
3952 m->llGuidsMachinesCreated.push_back(Guid(bstrNewMachineId));
3953
3954 LogFlowFuncLeave();
3955}
3956
3957/**
3958 * @throws HRESULT errors.
3959 */
3960void Appliance::i_importMachines(ImportStack &stack)
3961{
3962 // this is safe to access because this thread only gets started
3963 const ovf::OVFReader &reader = *m->pReader;
3964
3965 // create a session for the machine + disks we manipulate below
3966 HRESULT rc = stack.pSession.createInprocObject(CLSID_Session);
3967 ComAssertComRCThrowRC(rc);
3968
3969 list<ovf::VirtualSystem>::const_iterator it;
3970 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it1;
3971 /* Iterate through all virtual systems of that appliance */
3972 size_t i = 0;
3973 for (it = reader.m_llVirtualSystems.begin(), it1 = m->virtualSystemDescriptions.begin();
3974 it != reader.m_llVirtualSystems.end() && it1 != m->virtualSystemDescriptions.end();
3975 ++it, ++it1, ++i)
3976 {
3977 const ovf::VirtualSystem &vsysThis = *it;
3978 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it1);
3979
3980 ComPtr<IMachine> pNewMachine;
3981
3982 // there are two ways in which we can create a vbox machine from OVF:
3983 // -- either this OVF was written by vbox 3.2 or later, in which case there is a <vbox:Machine> element
3984 // in the <VirtualSystem>; then the VirtualSystemDescription::Data has a settings::MachineConfigFile
3985 // with all the machine config pretty-parsed;
3986 // -- or this is an OVF from an older vbox or an external source, and then we need to translate the
3987 // VirtualSystemDescriptionEntry and do import work
3988
3989 // Even for the vbox:Machine case, there are a number of configuration items that will be taken from
3990 // the OVF because otherwise the "override import parameters" mechanism in the GUI won't work.
3991
3992 // VM name
3993 std::list<VirtualSystemDescriptionEntry*> vsdeName = vsdescThis->i_findByType(VirtualSystemDescriptionType_Name);
3994 if (vsdeName.size() < 1)
3995 throw setError(VBOX_E_FILE_ERROR,
3996 tr("Missing VM name"));
3997 stack.strNameVBox = vsdeName.front()->strVBoxCurrent;
3998
3999 // have VirtualBox suggest where the filename would be placed so we can
4000 // put the disk images in the same directory
4001 Bstr bstrMachineFilename;
4002 rc = mVirtualBox->ComposeMachineFilename(Bstr(stack.strNameVBox).raw(),
4003 NULL /* aGroup */,
4004 NULL /* aCreateFlags */,
4005 NULL /* aBaseFolder */,
4006 bstrMachineFilename.asOutParam());
4007 if (FAILED(rc)) throw rc;
4008 // and determine the machine folder from that
4009 stack.strMachineFolder = bstrMachineFilename;
4010 stack.strMachineFolder.stripFilename();
4011 LogFunc(("i=%zu strName=%s bstrMachineFilename=%ls\n", i, stack.strNameVBox.c_str(), bstrMachineFilename.raw()));
4012
4013 // guest OS type
4014 std::list<VirtualSystemDescriptionEntry*> vsdeOS;
4015 vsdeOS = vsdescThis->i_findByType(VirtualSystemDescriptionType_OS);
4016 if (vsdeOS.size() < 1)
4017 throw setError(VBOX_E_FILE_ERROR,
4018 tr("Missing guest OS type"));
4019 stack.strOsTypeVBox = vsdeOS.front()->strVBoxCurrent;
4020
4021 // CPU count
4022 std::list<VirtualSystemDescriptionEntry*> vsdeCPU = vsdescThis->i_findByType(VirtualSystemDescriptionType_CPU);
4023 if (vsdeCPU.size() != 1)
4024 throw setError(VBOX_E_FILE_ERROR, tr("CPU count missing"));
4025
4026 stack.cCPUs = vsdeCPU.front()->strVBoxCurrent.toUInt32();
4027 // We need HWVirt & IO-APIC if more than one CPU is requested
4028 if (stack.cCPUs > 1)
4029 {
4030 stack.fForceHWVirt = true;
4031 stack.fForceIOAPIC = true;
4032 }
4033
4034 // RAM
4035 std::list<VirtualSystemDescriptionEntry*> vsdeRAM = vsdescThis->i_findByType(VirtualSystemDescriptionType_Memory);
4036 if (vsdeRAM.size() != 1)
4037 throw setError(VBOX_E_FILE_ERROR, tr("RAM size missing"));
4038 stack.ulMemorySizeMB = (ULONG)vsdeRAM.front()->strVBoxCurrent.toUInt64();
4039
4040#ifdef VBOX_WITH_USB
4041 // USB controller
4042 std::list<VirtualSystemDescriptionEntry*> vsdeUSBController =
4043 vsdescThis->i_findByType(VirtualSystemDescriptionType_USBController);
4044 // USB support is enabled if there's at least one such entry; to disable USB support,
4045 // the type of the USB item would have been changed to "ignore"
4046 stack.fUSBEnabled = !vsdeUSBController.empty();
4047#endif
4048 // audio adapter
4049 std::list<VirtualSystemDescriptionEntry*> vsdeAudioAdapter =
4050 vsdescThis->i_findByType(VirtualSystemDescriptionType_SoundCard);
4051 /* @todo: we support one audio adapter only */
4052 if (!vsdeAudioAdapter.empty())
4053 stack.strAudioAdapter = vsdeAudioAdapter.front()->strVBoxCurrent;
4054
4055 // for the description of the new machine, always use the OVF entry, the user may have changed it in the import config
4056 std::list<VirtualSystemDescriptionEntry*> vsdeDescription =
4057 vsdescThis->i_findByType(VirtualSystemDescriptionType_Description);
4058 if (!vsdeDescription.empty())
4059 stack.strDescription = vsdeDescription.front()->strVBoxCurrent;
4060
4061 // import vbox:machine or OVF now
4062 if (vsdescThis->m->pConfig)
4063 // vbox:Machine config
4064 i_importVBoxMachine(vsdescThis, pNewMachine, stack);
4065 else
4066 // generic OVF config
4067 i_importMachineGeneric(vsysThis, vsdescThis, pNewMachine, stack);
4068
4069 } // for (it = pAppliance->m->llVirtualSystems.begin() ...
4070}
4071
4072HRESULT Appliance::ImportStack::saveOriginalUUIDOfAttachedDevice(settings::AttachedDevice &device,
4073 const Utf8Str &newlyUuid)
4074{
4075 HRESULT rc = S_OK;
4076
4077 /* save for restoring */
4078 mapNewUUIDsToOriginalUUIDs.insert(std::make_pair(newlyUuid, device.uuid.toString()));
4079
4080 return rc;
4081}
4082
4083HRESULT Appliance::ImportStack::restoreOriginalUUIDOfAttachedDevice(settings::MachineConfigFile *config)
4084{
4085 HRESULT rc = S_OK;
4086
4087 settings::StorageControllersList &llControllers = config->storageMachine.llStorageControllers;
4088 settings::StorageControllersList::iterator itscl;
4089 for (itscl = llControllers.begin();
4090 itscl != llControllers.end();
4091 ++itscl)
4092 {
4093 settings::AttachedDevicesList &llAttachments = itscl->llAttachedDevices;
4094 settings::AttachedDevicesList::iterator itadl = llAttachments.begin();
4095 while (itadl != llAttachments.end())
4096 {
4097 std::map<Utf8Str , Utf8Str>::iterator it =
4098 mapNewUUIDsToOriginalUUIDs.find(itadl->uuid.toString());
4099 if(it!=mapNewUUIDsToOriginalUUIDs.end())
4100 {
4101 Utf8Str uuidOriginal = it->second;
4102 itadl->uuid = Guid(uuidOriginal);
4103 mapNewUUIDsToOriginalUUIDs.erase(it->first);
4104 }
4105 ++itadl;
4106 }
4107 }
4108
4109 return rc;
4110}
4111
4112/**
4113 * @throws Nothing
4114 */
4115RTVFSIOSTREAM Appliance::ImportStack::claimOvaLookAHead(void)
4116{
4117 RTVFSIOSTREAM hVfsIos = this->hVfsIosOvaLookAhead;
4118 this->hVfsIosOvaLookAhead = NIL_RTVFSIOSTREAM;
4119 /* We don't free the name since it may be referenced in error messages and such. */
4120 return hVfsIos;
4121}
4122
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