VirtualBox

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

Last change on this file since 63171 was 63171, checked in by vboxsync, 8 years ago

Main: warning about switches

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