VirtualBox

source: vbox/trunk/src/VBox/Main/src-client/ConsoleImpl2.cpp@ 54803

Last change on this file since 54803 was 54798, checked in by vboxsync, 10 years ago

Main/Console: Automatically resume the VM if all passwords were provided correctly

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 236.5 KB
Line 
1/* $Id: ConsoleImpl2.cpp 54798 2015-03-16 21:09:11Z vboxsync $ */
2/** @file
3 * VBox Console COM Class implementation - VM Configuration Bits.
4 *
5 * @remark We've split out the code that the 64-bit VC++ v8 compiler finds
6 * problematic to optimize so we can disable optimizations and later,
7 * perhaps, find a real solution for it (like rewriting the code and
8 * to stop resemble a tonne of spaghetti).
9 */
10
11/*
12 * Copyright (C) 2006-2014 Oracle Corporation
13 *
14 * This file is part of VirtualBox Open Source Edition (OSE), as
15 * available from http://www.virtualbox.org. This file is free software;
16 * you can redistribute it and/or modify it under the terms of the GNU
17 * General Public License (GPL) as published by the Free Software
18 * Foundation, in version 2 as it comes in the "COPYING" file of the
19 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
20 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
21 */
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26/* For some reason Windows burns in sdk\...\winsock.h if this isn't included first. */
27#include "VBox/com/ptr.h"
28
29#include "ConsoleImpl.h"
30#include "DisplayImpl.h"
31#ifdef VBOX_WITH_GUEST_CONTROL
32# include "GuestImpl.h"
33#endif
34#ifdef VBOX_WITH_DRAG_AND_DROP
35# include "GuestDnDPrivate.h"
36#endif
37#include "VMMDev.h"
38#include "Global.h"
39#ifdef VBOX_WITH_PCI_PASSTHROUGH
40# include "PCIRawDevImpl.h"
41#endif
42
43// generated header
44#include "SchemaDefs.h"
45
46#include "AutoCaller.h"
47#include "Logging.h"
48
49#include <iprt/base64.h>
50#include <iprt/buildconfig.h>
51#include <iprt/ctype.h>
52#include <iprt/dir.h>
53#include <iprt/file.h>
54#include <iprt/param.h>
55#include <iprt/path.h>
56#include <iprt/string.h>
57#include <iprt/system.h>
58#include <iprt/cpp/exception.h>
59#if 0 /* enable to play with lots of memory. */
60# include <iprt/env.h>
61#endif
62#include <iprt/stream.h>
63
64#include <VBox/vmm/vmapi.h>
65#include <VBox/err.h>
66#include <VBox/param.h>
67#include <VBox/vmm/pdmapi.h> /* For PDMR3DriverAttach/PDMR3DriverDetach. */
68#include <VBox/vmm/pdmusb.h> /* For PDMR3UsbCreateEmulatedDevice. */
69#include <VBox/version.h>
70#include <VBox/HostServices/VBoxClipboardSvc.h>
71#ifdef VBOX_WITH_CROGL
72# include <VBox/HostServices/VBoxCrOpenGLSvc.h>
73#include <VBox/VBoxOGL.h>
74#endif
75#ifdef VBOX_WITH_GUEST_PROPS
76# include <VBox/HostServices/GuestPropertySvc.h>
77# include <VBox/com/defs.h>
78# include <VBox/com/array.h>
79# include "HGCM.h" /** @todo It should be possible to register a service
80 * extension using a VMMDev callback. */
81# include <vector>
82#endif /* VBOX_WITH_GUEST_PROPS */
83#include <VBox/intnet.h>
84
85#include <VBox/com/com.h>
86#include <VBox/com/string.h>
87#include <VBox/com/array.h>
88
89#ifdef VBOX_WITH_NETFLT
90# if defined(RT_OS_SOLARIS)
91# include <zone.h>
92# elif defined(RT_OS_LINUX)
93# include <unistd.h>
94# include <sys/ioctl.h>
95# include <sys/socket.h>
96# include <linux/types.h>
97# include <linux/if.h>
98# include <linux/wireless.h>
99# elif defined(RT_OS_FREEBSD)
100# include <unistd.h>
101# include <sys/types.h>
102# include <sys/ioctl.h>
103# include <sys/socket.h>
104# include <net/if.h>
105# include <net80211/ieee80211_ioctl.h>
106# endif
107# if defined(RT_OS_WINDOWS)
108# include <VBox/VBoxNetCfg-win.h>
109# include <Ntddndis.h>
110# include <devguid.h>
111# else
112# include <HostNetworkInterfaceImpl.h>
113# include <netif.h>
114# include <stdlib.h>
115# endif
116#endif /* VBOX_WITH_NETFLT */
117
118#include "NetworkServiceRunner.h"
119#include "BusAssignmentManager.h"
120#ifdef VBOX_WITH_EXTPACK
121# include "ExtPackManagerImpl.h"
122#endif
123#if defined(RT_OS_DARWIN)
124# include "IOKit/IOKitLib.h"
125#endif
126
127
128/*******************************************************************************
129* Internal Functions *
130*******************************************************************************/
131static Utf8Str *GetExtraDataBoth(IVirtualBox *pVirtualBox, IMachine *pMachine, const char *pszName, Utf8Str *pStrValue);
132
133
134
135#if defined(RT_OS_DARWIN)
136
137static int DarwinSmcKey(char *pabKey, uint32_t cbKey)
138{
139 /*
140 * Method as described in Amit Singh's article:
141 * http://osxbook.com/book/bonus/chapter7/tpmdrmmyth/
142 */
143 typedef struct
144 {
145 uint32_t key;
146 uint8_t pad0[22];
147 uint32_t datasize;
148 uint8_t pad1[10];
149 uint8_t cmd;
150 uint32_t pad2;
151 uint8_t data[32];
152 } AppleSMCBuffer;
153
154 AssertReturn(cbKey >= 65, VERR_INTERNAL_ERROR);
155
156 io_service_t service = IOServiceGetMatchingService(kIOMasterPortDefault,
157 IOServiceMatching("AppleSMC"));
158 if (!service)
159 return VERR_NOT_FOUND;
160
161 io_connect_t port = (io_connect_t)0;
162 kern_return_t kr = IOServiceOpen(service, mach_task_self(), 0, &port);
163 IOObjectRelease(service);
164
165 if (kr != kIOReturnSuccess)
166 return RTErrConvertFromDarwin(kr);
167
168 AppleSMCBuffer inputStruct = { 0, {0}, 32, {0}, 5, };
169 AppleSMCBuffer outputStruct;
170 size_t cbOutputStruct = sizeof(outputStruct);
171
172 for (int i = 0; i < 2; i++)
173 {
174 inputStruct.key = (uint32_t)(i == 0 ? 'OSK0' : 'OSK1');
175 kr = IOConnectCallStructMethod((mach_port_t)port,
176 (uint32_t)2,
177 (const void *)&inputStruct,
178 sizeof(inputStruct),
179 (void *)&outputStruct,
180 &cbOutputStruct);
181 if (kr != kIOReturnSuccess)
182 {
183 IOServiceClose(port);
184 return RTErrConvertFromDarwin(kr);
185 }
186
187 for (int j = 0; j < 32; j++)
188 pabKey[j + i*32] = outputStruct.data[j];
189 }
190
191 IOServiceClose(port);
192
193 pabKey[64] = 0;
194
195 return VINF_SUCCESS;
196}
197
198#endif /* RT_OS_DARWIN */
199
200/* Darwin compile kludge */
201#undef PVM
202
203/* Comment out the following line to remove VMWare compatibility hack. */
204#define VMWARE_NET_IN_SLOT_11
205
206/**
207 * Translate IDE StorageControllerType_T to string representation.
208 */
209const char* controllerString(StorageControllerType_T enmType)
210{
211 switch (enmType)
212 {
213 case StorageControllerType_PIIX3:
214 return "PIIX3";
215 case StorageControllerType_PIIX4:
216 return "PIIX4";
217 case StorageControllerType_ICH6:
218 return "ICH6";
219 default:
220 return "Unknown";
221 }
222}
223
224/**
225 * Simple class for storing network boot information.
226 */
227struct BootNic
228{
229 ULONG mInstance;
230 PCIBusAddress mPCIAddress;
231
232 ULONG mBootPrio;
233 bool operator < (const BootNic &rhs) const
234 {
235 ULONG lval = mBootPrio - 1; /* 0 will wrap around and get the lowest priority. */
236 ULONG rval = rhs.mBootPrio - 1;
237 return lval < rval; /* Zero compares as highest number (lowest prio). */
238 }
239};
240
241static int findEfiRom(IVirtualBox* vbox, FirmwareType_T aFirmwareType, Utf8Str *pEfiRomFile)
242{
243 Bstr aFilePath, empty;
244 BOOL fPresent = FALSE;
245 HRESULT hrc = vbox->CheckFirmwarePresent(aFirmwareType, empty.raw(),
246 empty.asOutParam(), aFilePath.asOutParam(), &fPresent);
247 AssertComRCReturn(hrc, Global::vboxStatusCodeFromCOM(hrc));
248
249 if (!fPresent)
250 {
251 LogRel(("Failed to find an EFI ROM file.\n"));
252 return VERR_FILE_NOT_FOUND;
253 }
254
255 *pEfiRomFile = Utf8Str(aFilePath);
256
257 return VINF_SUCCESS;
258}
259
260/**
261 * @throws HRESULT on extra data retrival error.
262 */
263static int getSmcDeviceKey(IVirtualBox *pVirtualBox, IMachine *pMachine, Utf8Str *pStrKey, bool *pfGetKeyFromRealSMC)
264{
265 *pfGetKeyFromRealSMC = false;
266
267 /*
268 * The extra data takes precedence (if non-zero).
269 */
270 GetExtraDataBoth(pVirtualBox, pMachine, "VBoxInternal2/SmcDeviceKey", pStrKey);
271 if (pStrKey->isNotEmpty())
272 return VINF_SUCCESS;
273
274#ifdef RT_OS_DARWIN
275 /*
276 * Query it here and now.
277 */
278 char abKeyBuf[65];
279 int rc = DarwinSmcKey(abKeyBuf, sizeof(abKeyBuf));
280 if (SUCCEEDED(rc))
281 {
282 *pStrKey = abKeyBuf;
283 *pfGetKeyFromRealSMC = true;
284 return rc;
285 }
286 LogRel(("Warning: DarwinSmcKey failed with rc=%Rrc!\n", rc));
287
288#else
289 /*
290 * Is it apple hardware in bootcamp?
291 */
292 /** @todo implement + test RTSYSDMISTR_MANUFACTURER on all hosts.
293 * Currently falling back on the product name. */
294 char szManufacturer[256];
295 szManufacturer[0] = '\0';
296 RTSystemQueryDmiString(RTSYSDMISTR_MANUFACTURER, szManufacturer, sizeof(szManufacturer));
297 if (szManufacturer[0] != '\0')
298 {
299 if ( !strcmp(szManufacturer, "Apple Computer, Inc.")
300 || !strcmp(szManufacturer, "Apple Inc.")
301 )
302 *pfGetKeyFromRealSMC = true;
303 }
304 else
305 {
306 char szProdName[256];
307 szProdName[0] = '\0';
308 RTSystemQueryDmiString(RTSYSDMISTR_PRODUCT_NAME, szProdName, sizeof(szProdName));
309 if ( ( !strncmp(szProdName, RT_STR_TUPLE("Mac"))
310 || !strncmp(szProdName, RT_STR_TUPLE("iMac"))
311 || !strncmp(szProdName, RT_STR_TUPLE("iMac"))
312 || !strncmp(szProdName, RT_STR_TUPLE("Xserve"))
313 )
314 && !strchr(szProdName, ' ') /* no spaces */
315 && RT_C_IS_DIGIT(szProdName[strlen(szProdName) - 1]) /* version number */
316 )
317 *pfGetKeyFromRealSMC = true;
318 }
319
320 int rc = VINF_SUCCESS;
321#endif
322
323 return rc;
324}
325
326
327/*
328 * VC++ 8 / amd64 has some serious trouble with the next functions.
329 * As a temporary measure, we'll drop global optimizations.
330 */
331#if defined(_MSC_VER) && defined(RT_ARCH_AMD64)
332# pragma optimize("g", off)
333#endif
334
335static const char *const g_apszIDEDrives[4] =
336 { "PrimaryMaster", "PrimarySlave", "SecondaryMaster", "SecondarySlave" };
337
338class ConfigError : public RTCError
339{
340public:
341
342 ConfigError(const char *pcszFunction,
343 int vrc,
344 const char *pcszName)
345 : RTCError(Utf8StrFmt("%s failed: rc=%Rrc, pcszName=%s", pcszFunction, vrc, pcszName)),
346 m_vrc(vrc)
347 {
348 AssertMsgFailed(("%s\n", what())); // in strict mode, hit a breakpoint here
349 }
350
351 int m_vrc;
352};
353
354
355/**
356 * Helper that calls CFGMR3InsertString and throws an RTCError if that
357 * fails (C-string variant).
358 * @param pParent See CFGMR3InsertStringN.
359 * @param pcszNodeName See CFGMR3InsertStringN.
360 * @param pcszValue The string value.
361 */
362static void InsertConfigString(PCFGMNODE pNode,
363 const char *pcszName,
364 const char *pcszValue)
365{
366 int vrc = CFGMR3InsertString(pNode,
367 pcszName,
368 pcszValue);
369 if (RT_FAILURE(vrc))
370 throw ConfigError("CFGMR3InsertString", vrc, pcszName);
371}
372
373/**
374 * Helper that calls CFGMR3InsertString and throws an RTCError if that
375 * fails (Utf8Str variant).
376 * @param pParent See CFGMR3InsertStringN.
377 * @param pcszNodeName See CFGMR3InsertStringN.
378 * @param rStrValue The string value.
379 */
380static void InsertConfigString(PCFGMNODE pNode,
381 const char *pcszName,
382 const Utf8Str &rStrValue)
383{
384 int vrc = CFGMR3InsertStringN(pNode,
385 pcszName,
386 rStrValue.c_str(),
387 rStrValue.length());
388 if (RT_FAILURE(vrc))
389 throw ConfigError("CFGMR3InsertStringLengthKnown", vrc, pcszName);
390}
391
392/**
393 * Helper that calls CFGMR3InsertString and throws an RTCError if that
394 * fails (Bstr variant).
395 *
396 * @param pParent See CFGMR3InsertStringN.
397 * @param pcszNodeName See CFGMR3InsertStringN.
398 * @param rBstrValue The string value.
399 */
400static void InsertConfigString(PCFGMNODE pNode,
401 const char *pcszName,
402 const Bstr &rBstrValue)
403{
404 InsertConfigString(pNode, pcszName, Utf8Str(rBstrValue));
405}
406
407/**
408 * Helper that calls CFGMR3InsertBytes and throws an RTCError if that fails.
409 *
410 * @param pNode See CFGMR3InsertBytes.
411 * @param pcszName See CFGMR3InsertBytes.
412 * @param pvBytes See CFGMR3InsertBytes.
413 * @param cbBytes See CFGMR3InsertBytes.
414 */
415static void InsertConfigBytes(PCFGMNODE pNode,
416 const char *pcszName,
417 const void *pvBytes,
418 size_t cbBytes)
419{
420 int vrc = CFGMR3InsertBytes(pNode,
421 pcszName,
422 pvBytes,
423 cbBytes);
424 if (RT_FAILURE(vrc))
425 throw ConfigError("CFGMR3InsertBytes", vrc, pcszName);
426}
427
428/**
429 * Helper that calls CFGMR3InsertInteger and throws an RTCError if that
430 * fails.
431 *
432 * @param pNode See CFGMR3InsertInteger.
433 * @param pcszName See CFGMR3InsertInteger.
434 * @param u64Integer See CFGMR3InsertInteger.
435 */
436static void InsertConfigInteger(PCFGMNODE pNode,
437 const char *pcszName,
438 uint64_t u64Integer)
439{
440 int vrc = CFGMR3InsertInteger(pNode,
441 pcszName,
442 u64Integer);
443 if (RT_FAILURE(vrc))
444 throw ConfigError("CFGMR3InsertInteger", vrc, pcszName);
445}
446
447/**
448 * Helper that calls CFGMR3InsertNode and throws an RTCError if that fails.
449 *
450 * @param pNode See CFGMR3InsertNode.
451 * @param pcszName See CFGMR3InsertNode.
452 * @param ppChild See CFGMR3InsertNode.
453 */
454static void InsertConfigNode(PCFGMNODE pNode,
455 const char *pcszName,
456 PCFGMNODE *ppChild)
457{
458 int vrc = CFGMR3InsertNode(pNode, pcszName, ppChild);
459 if (RT_FAILURE(vrc))
460 throw ConfigError("CFGMR3InsertNode", vrc, pcszName);
461}
462
463/**
464 * Helper that calls CFGMR3RemoveValue and throws an RTCError if that fails.
465 *
466 * @param pNode See CFGMR3RemoveValue.
467 * @param pcszName See CFGMR3RemoveValue.
468 */
469static void RemoveConfigValue(PCFGMNODE pNode,
470 const char *pcszName)
471{
472 int vrc = CFGMR3RemoveValue(pNode, pcszName);
473 if (RT_FAILURE(vrc))
474 throw ConfigError("CFGMR3RemoveValue", vrc, pcszName);
475}
476
477/**
478 * Gets an extra data value, consulting both machine and global extra data.
479 *
480 * @throws HRESULT on failure
481 * @returns pStrValue for the callers convenience.
482 * @param pVirtualBox Pointer to the IVirtualBox interface.
483 * @param pMachine Pointer to the IMachine interface.
484 * @param pszName The value to get.
485 * @param pStrValue Where to return it's value (empty string if not
486 * found).
487 */
488static Utf8Str *GetExtraDataBoth(IVirtualBox *pVirtualBox, IMachine *pMachine, const char *pszName, Utf8Str *pStrValue)
489{
490 pStrValue->setNull();
491
492 Bstr bstrName(pszName);
493 Bstr bstrValue;
494 HRESULT hrc = pMachine->GetExtraData(bstrName.raw(), bstrValue.asOutParam());
495 if (FAILED(hrc))
496 throw hrc;
497 if (bstrValue.isEmpty())
498 {
499 hrc = pVirtualBox->GetExtraData(bstrName.raw(), bstrValue.asOutParam());
500 if (FAILED(hrc))
501 throw hrc;
502 }
503
504 if (bstrValue.isNotEmpty())
505 *pStrValue = bstrValue;
506 return pStrValue;
507}
508
509
510/** Helper that finds out the next HBA port used
511 */
512static LONG GetNextUsedPort(LONG aPortUsed[30], LONG lBaseVal, uint32_t u32Size)
513{
514 LONG lNextPortUsed = 30;
515 for (size_t j = 0; j < u32Size; ++j)
516 {
517 if ( aPortUsed[j] > lBaseVal
518 && aPortUsed[j] <= lNextPortUsed)
519 lNextPortUsed = aPortUsed[j];
520 }
521 return lNextPortUsed;
522}
523
524#define MAX_BIOS_LUN_COUNT 4
525
526static int SetBiosDiskInfo(ComPtr<IMachine> pMachine, PCFGMNODE pCfg, PCFGMNODE pBiosCfg,
527 Bstr controllerName, const char * const s_apszBiosConfig[4])
528{
529 HRESULT hrc;
530#define MAX_DEVICES 30
531#define H() AssertLogRelMsgReturn(!FAILED(hrc), ("hrc=%Rhrc\n", hrc), VERR_MAIN_CONFIG_CONSTRUCTOR_COM_ERROR)
532
533 LONG lPortLUN[MAX_BIOS_LUN_COUNT];
534 LONG lPortUsed[MAX_DEVICES];
535 uint32_t u32HDCount = 0;
536
537 /* init to max value */
538 lPortLUN[0] = MAX_DEVICES;
539
540 com::SafeIfaceArray<IMediumAttachment> atts;
541 hrc = pMachine->GetMediumAttachmentsOfController(controllerName.raw(),
542 ComSafeArrayAsOutParam(atts)); H();
543 size_t uNumAttachments = atts.size();
544 if (uNumAttachments > MAX_DEVICES)
545 {
546 LogRel(("Number of Attachments > Max=%d.\n", uNumAttachments));
547 uNumAttachments = MAX_DEVICES;
548 }
549
550 /* Find the relevant ports/IDs, i.e the ones to which a HD is attached. */
551 for (size_t j = 0; j < uNumAttachments; ++j)
552 {
553 IMediumAttachment *pMediumAtt = atts[j];
554 LONG lPortNum = 0;
555 hrc = pMediumAtt->COMGETTER(Port)(&lPortNum); H();
556 if (SUCCEEDED(hrc))
557 {
558 DeviceType_T lType;
559 hrc = pMediumAtt->COMGETTER(Type)(&lType); H();
560 if (SUCCEEDED(hrc) && lType == DeviceType_HardDisk)
561 {
562 /* find min port number used for HD */
563 if (lPortNum < lPortLUN[0])
564 lPortLUN[0] = lPortNum;
565 lPortUsed[u32HDCount++] = lPortNum;
566 LogFlowFunc(("HD port Count=%d\n", u32HDCount));
567 }
568
569 /* Configure the hotpluggable flag for the port. */
570 BOOL fHotPluggable = FALSE;
571 hrc = pMediumAtt->COMGETTER(HotPluggable)(&fHotPluggable); H();
572 if (SUCCEEDED(hrc))
573 {
574 PCFGMNODE pPortCfg;
575 char szName[24];
576 RTStrPrintf(szName, sizeof(szName), "Port%d", lPortNum);
577
578 InsertConfigNode(pCfg, szName, &pPortCfg);
579 InsertConfigInteger(pPortCfg, "Hotpluggable", fHotPluggable ? 1 : 0);
580 }
581 }
582 }
583
584
585 /* Pick only the top 4 used HD Ports as CMOS doesn't have space
586 * to save details for all 30 ports
587 */
588 uint32_t u32MaxPortCount = MAX_BIOS_LUN_COUNT;
589 if (u32HDCount < MAX_BIOS_LUN_COUNT)
590 u32MaxPortCount = u32HDCount;
591 for (size_t j = 1; j < u32MaxPortCount; j++)
592 lPortLUN[j] = GetNextUsedPort(lPortUsed,
593 lPortLUN[j-1],
594 u32HDCount);
595 if (pBiosCfg)
596 {
597 for (size_t j = 0; j < u32MaxPortCount; j++)
598 {
599 InsertConfigInteger(pBiosCfg, s_apszBiosConfig[j], lPortLUN[j]);
600 LogFlowFunc(("Top %d HBA ports = %s, %d\n", j, s_apszBiosConfig[j], lPortLUN[j]));
601 }
602 }
603 return VINF_SUCCESS;
604}
605
606#ifdef VBOX_WITH_PCI_PASSTHROUGH
607HRESULT Console::i_attachRawPCIDevices(PUVM pUVM, BusAssignmentManager *pBusMgr, PCFGMNODE pDevices)
608{
609 HRESULT hrc = S_OK;
610 PCFGMNODE pInst, pCfg, pLunL0, pLunL1;
611
612 SafeIfaceArray<IPCIDeviceAttachment> assignments;
613 ComPtr<IMachine> aMachine = i_machine();
614
615 hrc = aMachine->COMGETTER(PCIDeviceAssignments)(ComSafeArrayAsOutParam(assignments));
616 if ( hrc != S_OK
617 || assignments.size() < 1)
618 return hrc;
619
620 /*
621 * PCI passthrough is only available if the proper ExtPack is installed.
622 *
623 * Note. Configuring PCI passthrough here and providing messages about
624 * the missing extpack isn't exactly clean, but it is a necessary evil
625 * to patch over legacy compatability issues introduced by the new
626 * distribution model.
627 */
628# ifdef VBOX_WITH_EXTPACK
629 static const char *s_pszPCIRawExtPackName = "Oracle VM VirtualBox Extension Pack";
630 if (!mptrExtPackManager->i_isExtPackUsable(s_pszPCIRawExtPackName))
631 /* Always fatal! */
632 return VMR3SetError(pUVM, VERR_NOT_FOUND, RT_SRC_POS,
633 N_("Implementation of the PCI passthrough framework not found!\n"
634 "The VM cannot be started. To fix this problem, either "
635 "install the '%s' or disable PCI passthrough via VBoxManage"),
636 s_pszPCIRawExtPackName);
637# endif
638
639 PCFGMNODE pBridges = CFGMR3GetChild(pDevices, "ich9pcibridge");
640 Assert(pBridges);
641
642 /* Find required bridges, and add missing ones */
643 for (size_t iDev = 0; iDev < assignments.size(); iDev++)
644 {
645 ComPtr<IPCIDeviceAttachment> assignment = assignments[iDev];
646 LONG guest = 0;
647 PCIBusAddress GuestPCIAddress;
648
649 assignment->COMGETTER(GuestAddress)(&guest);
650 GuestPCIAddress.fromLong(guest);
651 Assert(GuestPCIAddress.valid());
652
653 if (GuestPCIAddress.miBus > 0)
654 {
655 int iBridgesMissed = 0;
656 int iBase = GuestPCIAddress.miBus - 1;
657
658 while (!pBusMgr->hasPCIDevice("ich9pcibridge", iBase) && iBase > 0)
659 {
660 iBridgesMissed++; iBase--;
661 }
662 iBase++;
663
664 for (int iBridge = 0; iBridge < iBridgesMissed; iBridge++)
665 {
666 InsertConfigNode(pBridges, Utf8StrFmt("%d", iBase + iBridge).c_str(), &pInst);
667 InsertConfigInteger(pInst, "Trusted", 1);
668 hrc = pBusMgr->assignPCIDevice("ich9pcibridge", pInst);
669 }
670 }
671 }
672
673 /* Now actually add devices */
674 PCFGMNODE pPCIDevs = NULL;
675
676 if (assignments.size() > 0)
677 {
678 InsertConfigNode(pDevices, "pciraw", &pPCIDevs);
679
680 PCFGMNODE pRoot = CFGMR3GetParent(pDevices); Assert(pRoot);
681
682 /* Tell PGM to tell GPCIRaw about guest mappings. */
683 CFGMR3InsertNode(pRoot, "PGM", NULL);
684 InsertConfigInteger(CFGMR3GetChild(pRoot, "PGM"), "PciPassThrough", 1);
685
686 /*
687 * Currently, using IOMMU needed for PCI passthrough
688 * requires RAM preallocation.
689 */
690 /** @todo: check if we can lift this requirement */
691 CFGMR3RemoveValue(pRoot, "RamPreAlloc");
692 InsertConfigInteger(pRoot, "RamPreAlloc", 1);
693 }
694
695 for (size_t iDev = 0; iDev < assignments.size(); iDev++)
696 {
697 PCIBusAddress HostPCIAddress, GuestPCIAddress;
698 ComPtr<IPCIDeviceAttachment> assignment = assignments[iDev];
699 LONG host, guest;
700 Bstr aDevName;
701
702 assignment->COMGETTER(HostAddress)(&host);
703 assignment->COMGETTER(GuestAddress)(&guest);
704 assignment->COMGETTER(Name)(aDevName.asOutParam());
705
706 InsertConfigNode(pPCIDevs, Utf8StrFmt("%d", iDev).c_str(), &pInst);
707 InsertConfigInteger(pInst, "Trusted", 1);
708
709 HostPCIAddress.fromLong(host);
710 Assert(HostPCIAddress.valid());
711 InsertConfigNode(pInst, "Config", &pCfg);
712 InsertConfigString(pCfg, "DeviceName", aDevName);
713
714 InsertConfigInteger(pCfg, "DetachHostDriver", 1);
715 InsertConfigInteger(pCfg, "HostPCIBusNo", HostPCIAddress.miBus);
716 InsertConfigInteger(pCfg, "HostPCIDeviceNo", HostPCIAddress.miDevice);
717 InsertConfigInteger(pCfg, "HostPCIFunctionNo", HostPCIAddress.miFn);
718
719 GuestPCIAddress.fromLong(guest);
720 Assert(GuestPCIAddress.valid());
721 hrc = pBusMgr->assignHostPCIDevice("pciraw", pInst, HostPCIAddress, GuestPCIAddress, true);
722 if (hrc != S_OK)
723 return hrc;
724
725 InsertConfigInteger(pCfg, "GuestPCIBusNo", GuestPCIAddress.miBus);
726 InsertConfigInteger(pCfg, "GuestPCIDeviceNo", GuestPCIAddress.miDevice);
727 InsertConfigInteger(pCfg, "GuestPCIFunctionNo", GuestPCIAddress.miFn);
728
729 /* the driver */
730 InsertConfigNode(pInst, "LUN#0", &pLunL0);
731 InsertConfigString(pLunL0, "Driver", "pciraw");
732 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
733
734 /* the Main driver */
735 InsertConfigString(pLunL1, "Driver", "MainPciRaw");
736 InsertConfigNode(pLunL1, "Config", &pCfg);
737 PCIRawDev* pMainDev = new PCIRawDev(this);
738 InsertConfigInteger(pCfg, "Object", (uintptr_t)pMainDev);
739 }
740
741 return hrc;
742}
743#endif
744
745
746void Console::i_attachStatusDriver(PCFGMNODE pCtlInst, PPDMLED *papLeds,
747 uint64_t uFirst, uint64_t uLast,
748 Console::MediumAttachmentMap *pmapMediumAttachments,
749 const char *pcszDevice, unsigned uInstance)
750{
751 PCFGMNODE pLunL0, pCfg;
752 InsertConfigNode(pCtlInst, "LUN#999", &pLunL0);
753 InsertConfigString(pLunL0, "Driver", "MainStatus");
754 InsertConfigNode(pLunL0, "Config", &pCfg);
755 InsertConfigInteger(pCfg, "papLeds", (uintptr_t)papLeds);
756 if (pmapMediumAttachments)
757 {
758 InsertConfigInteger(pCfg, "pmapMediumAttachments", (uintptr_t)pmapMediumAttachments);
759 InsertConfigInteger(pCfg, "pConsole", (uintptr_t)this);
760 AssertPtr(pcszDevice);
761 Utf8Str deviceInstance = Utf8StrFmt("%s/%u", pcszDevice, uInstance);
762 InsertConfigString(pCfg, "DeviceInstance", deviceInstance.c_str());
763 }
764 InsertConfigInteger(pCfg, "First", uFirst);
765 InsertConfigInteger(pCfg, "Last", uLast);
766}
767
768
769/**
770 * Construct the VM configuration tree (CFGM).
771 *
772 * This is a callback for VMR3Create() call. It is called from CFGMR3Init()
773 * in the emulation thread (EMT). Any per thread COM/XPCOM initialization
774 * is done here.
775 *
776 * @param pUVM The user mode VM handle.
777 * @param pVM The cross context VM handle.
778 * @param pvConsole Pointer to the VMPowerUpTask object.
779 * @return VBox status code.
780 *
781 * @note Locks the Console object for writing.
782 */
783DECLCALLBACK(int) Console::i_configConstructor(PUVM pUVM, PVM pVM, void *pvConsole)
784{
785 LogFlowFuncEnter();
786
787 AssertReturn(pvConsole, VERR_INVALID_POINTER);
788 ComObjPtr<Console> pConsole = static_cast<Console *>(pvConsole);
789
790 AutoCaller autoCaller(pConsole);
791 AssertComRCReturn(autoCaller.rc(), VERR_ACCESS_DENIED);
792
793 /* lock the console because we widely use internal fields and methods */
794 AutoWriteLock alock(pConsole COMMA_LOCKVAL_SRC_POS);
795
796 /*
797 * Set the VM handle and do the rest of the job in an worker method so we
798 * can easily reset the VM handle on failure.
799 */
800 pConsole->mpUVM = pUVM;
801 VMR3RetainUVM(pUVM);
802 int vrc;
803 try
804 {
805 vrc = pConsole->i_configConstructorInner(pUVM, pVM, &alock);
806 }
807 catch (...)
808 {
809 vrc = VERR_UNEXPECTED_EXCEPTION;
810 }
811 if (RT_FAILURE(vrc))
812 {
813 pConsole->mpUVM = NULL;
814 VMR3ReleaseUVM(pUVM);
815 }
816
817 return vrc;
818}
819
820
821/**
822 * Worker for configConstructor.
823 *
824 * @return VBox status code.
825 * @param pUVM The user mode VM handle.
826 * @param pVM The cross context VM handle.
827 * @param pAlock The automatic lock instance. This is for when we have
828 * to leave it in order to avoid deadlocks (ext packs and
829 * more).
830 */
831int Console::i_configConstructorInner(PUVM pUVM, PVM pVM, AutoWriteLock *pAlock)
832{
833 VMMDev *pVMMDev = m_pVMMDev; Assert(pVMMDev);
834 ComPtr<IMachine> pMachine = i_machine();
835
836 int rc;
837 HRESULT hrc;
838 Utf8Str strTmp;
839 Bstr bstr;
840
841#define H() AssertLogRelMsgReturn(!FAILED(hrc), ("hrc=%Rhrc\n", hrc), VERR_MAIN_CONFIG_CONSTRUCTOR_COM_ERROR)
842
843 /*
844 * Get necessary objects and frequently used parameters.
845 */
846 ComPtr<IVirtualBox> virtualBox;
847 hrc = pMachine->COMGETTER(Parent)(virtualBox.asOutParam()); H();
848
849 ComPtr<IHost> host;
850 hrc = virtualBox->COMGETTER(Host)(host.asOutParam()); H();
851
852 ComPtr<ISystemProperties> systemProperties;
853 hrc = virtualBox->COMGETTER(SystemProperties)(systemProperties.asOutParam()); H();
854
855 ComPtr<IBIOSSettings> biosSettings;
856 hrc = pMachine->COMGETTER(BIOSSettings)(biosSettings.asOutParam()); H();
857
858 hrc = pMachine->COMGETTER(HardwareUUID)(bstr.asOutParam()); H();
859 RTUUID HardwareUuid;
860 rc = RTUuidFromUtf16(&HardwareUuid, bstr.raw());
861 AssertRCReturn(rc, rc);
862
863 ULONG cRamMBs;
864 hrc = pMachine->COMGETTER(MemorySize)(&cRamMBs); H();
865#if 0 /* enable to play with lots of memory. */
866 if (RTEnvExist("VBOX_RAM_SIZE"))
867 cRamMBs = RTStrToUInt64(RTEnvGet("VBOX_RAM_SIZE"));
868#endif
869 uint64_t const cbRam = cRamMBs * (uint64_t)_1M;
870 uint32_t cbRamHole = MM_RAM_HOLE_SIZE_DEFAULT;
871 uint64_t uMcfgBase = 0;
872 uint32_t cbMcfgLength = 0;
873
874 ParavirtProvider_T paravirtProvider;
875 hrc = pMachine->GetEffectiveParavirtProvider(&paravirtProvider); H();
876
877 ChipsetType_T chipsetType;
878 hrc = pMachine->COMGETTER(ChipsetType)(&chipsetType); H();
879 if (chipsetType == ChipsetType_ICH9)
880 {
881 /* We'd better have 0x10000000 region, to cover 256 buses
882 but this put too much load on hypervisor heap */
883 cbMcfgLength = 0x4000000; //0x10000000;
884 cbRamHole += cbMcfgLength;
885 uMcfgBase = _4G - cbRamHole;
886 }
887
888 BusAssignmentManager *pBusMgr = mBusMgr = BusAssignmentManager::createInstance(chipsetType);
889
890 ULONG cCpus = 1;
891 hrc = pMachine->COMGETTER(CPUCount)(&cCpus); H();
892
893 ULONG ulCpuExecutionCap = 100;
894 hrc = pMachine->COMGETTER(CPUExecutionCap)(&ulCpuExecutionCap); H();
895
896 Bstr osTypeId;
897 hrc = pMachine->COMGETTER(OSTypeId)(osTypeId.asOutParam()); H();
898 LogRel(("Guest OS type: '%s'\n", Utf8Str(osTypeId).c_str()));
899
900 BOOL fIOAPIC;
901 hrc = biosSettings->COMGETTER(IOAPICEnabled)(&fIOAPIC); H();
902
903 ComPtr<IGuestOSType> guestOSType;
904 hrc = virtualBox->GetGuestOSType(osTypeId.raw(), guestOSType.asOutParam()); H();
905
906 Bstr guestTypeFamilyId;
907 hrc = guestOSType->COMGETTER(FamilyId)(guestTypeFamilyId.asOutParam()); H();
908 BOOL fOsXGuest = guestTypeFamilyId == Bstr("MacOS");
909
910 ULONG maxNetworkAdapters;
911 hrc = systemProperties->GetMaxNetworkAdapters(chipsetType, &maxNetworkAdapters); H();
912 /*
913 * Get root node first.
914 * This is the only node in the tree.
915 */
916 PCFGMNODE pRoot = CFGMR3GetRootU(pUVM);
917 Assert(pRoot);
918
919 // InsertConfigString throws
920 try
921 {
922
923 /*
924 * Set the root (and VMM) level values.
925 */
926 hrc = pMachine->COMGETTER(Name)(bstr.asOutParam()); H();
927 InsertConfigString(pRoot, "Name", bstr);
928 InsertConfigBytes(pRoot, "UUID", &HardwareUuid, sizeof(HardwareUuid));
929 InsertConfigInteger(pRoot, "RamSize", cbRam);
930 InsertConfigInteger(pRoot, "RamHoleSize", cbRamHole);
931 InsertConfigInteger(pRoot, "NumCPUs", cCpus);
932 InsertConfigInteger(pRoot, "CpuExecutionCap", ulCpuExecutionCap);
933 InsertConfigInteger(pRoot, "TimerMillies", 10);
934#ifdef VBOX_WITH_RAW_MODE
935 InsertConfigInteger(pRoot, "RawR3Enabled", 1); /* boolean */
936 InsertConfigInteger(pRoot, "RawR0Enabled", 1); /* boolean */
937 /** @todo Config: RawR0, PATMEnabled and CSAMEnabled needs attention later. */
938 InsertConfigInteger(pRoot, "PATMEnabled", 1); /* boolean */
939 InsertConfigInteger(pRoot, "CSAMEnabled", 1); /* boolean */
940#endif
941
942#ifdef VBOX_WITH_RAW_RING1
943 if (osTypeId == "QNX")
944 {
945 /* QNX needs special treatment in raw mode due to its use of ring-1. */
946 InsertConfigInteger(pRoot, "RawR1Enabled", 1); /* boolean */
947 }
948#endif
949
950 /* Not necessary, but to make sure these two settings end up in the release log. */
951 BOOL fPageFusion = FALSE;
952 hrc = pMachine->COMGETTER(PageFusionEnabled)(&fPageFusion); H();
953 InsertConfigInteger(pRoot, "PageFusion", fPageFusion); /* boolean */
954 ULONG ulBalloonSize = 0;
955 hrc = pMachine->COMGETTER(MemoryBalloonSize)(&ulBalloonSize); H();
956 InsertConfigInteger(pRoot, "MemBalloonSize", ulBalloonSize);
957
958 /*
959 * CPUM values.
960 */
961 PCFGMNODE pCPUM;
962 InsertConfigNode(pRoot, "CPUM", &pCPUM);
963
964 /* cpuid leaf overrides. */
965 static uint32_t const s_auCpuIdRanges[] =
966 {
967 UINT32_C(0x00000000), UINT32_C(0x0000000a),
968 UINT32_C(0x80000000), UINT32_C(0x8000000a)
969 };
970 for (unsigned i = 0; i < RT_ELEMENTS(s_auCpuIdRanges); i += 2)
971 for (uint32_t uLeaf = s_auCpuIdRanges[i]; uLeaf < s_auCpuIdRanges[i + 1]; uLeaf++)
972 {
973 ULONG ulEax, ulEbx, ulEcx, ulEdx;
974 hrc = pMachine->GetCPUIDLeaf(uLeaf, &ulEax, &ulEbx, &ulEcx, &ulEdx);
975 if (SUCCEEDED(hrc))
976 {
977 PCFGMNODE pLeaf;
978 InsertConfigNode(pCPUM, Utf8StrFmt("HostCPUID/%RX32", uLeaf).c_str(), &pLeaf);
979
980 InsertConfigInteger(pLeaf, "eax", ulEax);
981 InsertConfigInteger(pLeaf, "ebx", ulEbx);
982 InsertConfigInteger(pLeaf, "ecx", ulEcx);
983 InsertConfigInteger(pLeaf, "edx", ulEdx);
984 }
985 else if (hrc != E_INVALIDARG) H();
986 }
987
988 /* We must limit CPUID count for Windows NT 4, as otherwise it stops
989 with error 0x3e (MULTIPROCESSOR_CONFIGURATION_NOT_SUPPORTED). */
990 if (osTypeId == "WindowsNT4")
991 {
992 LogRel(("Limiting CPUID leaf count for NT4 guests\n"));
993 InsertConfigInteger(pCPUM, "NT4LeafLimit", true);
994 }
995
996 /* Expose CMPXCHG16B. Currently a hack. */
997 if ( osTypeId == "Windows81_64"
998 || osTypeId == "Windows2012_64"
999 || osTypeId == "Windows10_64")
1000 {
1001 LogRel(("Enabling CMPXCHG16B for Windows 8.1 / 2k12 or newer guests\n"));
1002 InsertConfigInteger(pCPUM, "CMPXCHG16B", true);
1003 }
1004
1005 if (fOsXGuest)
1006 {
1007 /* Expose extended MWAIT features to Mac OS X guests. */
1008 LogRel(("Using MWAIT extensions\n"));
1009 InsertConfigInteger(pCPUM, "MWaitExtensions", true);
1010
1011 /* Fake the CPU family/model so the guest works. This is partly
1012 because older mac releases really doesn't work on newer cpus,
1013 and partly because mac os x expects more from systems with newer
1014 cpus (MSRs, power features, whatever). */
1015 uint32_t uMaxIntelFamilyModelStep = UINT32_MAX;
1016 if ( osTypeId == "MacOS"
1017 || osTypeId == "MacOS_64")
1018 uMaxIntelFamilyModelStep = RT_MAKE_U32_FROM_U8(1, 23, 6, 7); /* Penryn / X5482. */
1019 else if ( osTypeId == "MacOS106"
1020 || osTypeId == "MacOS106_64")
1021 uMaxIntelFamilyModelStep = RT_MAKE_U32_FROM_U8(1, 23, 6, 7); /* Penryn / X5482 */
1022 else if ( osTypeId == "MacOS107"
1023 || osTypeId == "MacOS107_64")
1024 uMaxIntelFamilyModelStep = RT_MAKE_U32_FROM_U8(1, 23, 6, 7); /* Penryn / X5482 */ /** @todo figure out
1025 what is required here. */
1026 else if ( osTypeId == "MacOS108"
1027 || osTypeId == "MacOS108_64")
1028 uMaxIntelFamilyModelStep = RT_MAKE_U32_FROM_U8(1, 23, 6, 7); /* Penryn / X5482 */ /** @todo figure out
1029 what is required here. */
1030 else if ( osTypeId == "MacOS109"
1031 || osTypeId == "MacOS109_64")
1032 uMaxIntelFamilyModelStep = RT_MAKE_U32_FROM_U8(1, 23, 6, 7); /* Penryn / X5482 */ /** @todo figure
1033 out what is required here. */
1034 if (uMaxIntelFamilyModelStep != UINT32_MAX)
1035 InsertConfigInteger(pCPUM, "MaxIntelFamilyModelStep", uMaxIntelFamilyModelStep);
1036 }
1037
1038 /* Synthetic CPU */
1039 BOOL fSyntheticCpu = false;
1040 hrc = pMachine->GetCPUProperty(CPUPropertyType_Synthetic, &fSyntheticCpu); H();
1041 InsertConfigInteger(pCPUM, "SyntheticCpu", fSyntheticCpu);
1042
1043 /* Physical Address Extension (PAE) */
1044 BOOL fEnablePAE = false;
1045 hrc = pMachine->GetCPUProperty(CPUPropertyType_PAE, &fEnablePAE); H();
1046 InsertConfigInteger(pRoot, "EnablePAE", fEnablePAE);
1047
1048
1049 /*
1050 * Hardware virtualization extensions.
1051 */
1052 BOOL fSupportsHwVirtEx;
1053 hrc = host->GetProcessorFeature(ProcessorFeature_HWVirtEx, &fSupportsHwVirtEx); H();
1054
1055 BOOL fIsGuest64Bit;
1056 hrc = pMachine->GetCPUProperty(CPUPropertyType_LongMode, &fIsGuest64Bit); H();
1057 if (fIsGuest64Bit)
1058 {
1059 BOOL fSupportsLongMode;
1060 hrc = host->GetProcessorFeature(ProcessorFeature_LongMode, &fSupportsLongMode); H();
1061 if (!fSupportsLongMode)
1062 {
1063 LogRel(("WARNING! 64-bit guest type selected but the host CPU does NOT support 64-bit.\n"));
1064 fIsGuest64Bit = FALSE;
1065 }
1066 if (!fSupportsHwVirtEx)
1067 {
1068 LogRel(("WARNING! 64-bit guest type selected but the host CPU does NOT support HW virtualization.\n"));
1069 fIsGuest64Bit = FALSE;
1070 }
1071 }
1072
1073 BOOL fHMEnabled;
1074 hrc = pMachine->GetHWVirtExProperty(HWVirtExPropertyType_Enabled, &fHMEnabled); H();
1075 if (cCpus > 1 && !fHMEnabled)
1076 {
1077 LogRel(("Forced fHMEnabled to TRUE by SMP guest.\n"));
1078 fHMEnabled = TRUE;
1079 }
1080
1081 BOOL fHMForced;
1082#ifdef VBOX_WITH_RAW_MODE
1083 /* - With more than 4GB PGM will use different RAMRANGE sizes for raw
1084 mode and hv mode to optimize lookup times.
1085 - With more than one virtual CPU, raw-mode isn't a fallback option.
1086 - With a 64-bit guest, raw-mode isn't a fallback option either. */
1087 fHMForced = fHMEnabled
1088 && ( cbRam + cbRamHole > _4G
1089 || cCpus > 1
1090 || fIsGuest64Bit);
1091# ifdef RT_OS_DARWIN
1092 fHMForced = fHMEnabled;
1093# endif
1094 if (fHMForced)
1095 {
1096 if (cbRam + cbRamHole > _4G)
1097 LogRel(("fHMForced=true - Lots of RAM\n"));
1098 if (cCpus > 1)
1099 LogRel(("fHMForced=true - SMP\n"));
1100 if (fIsGuest64Bit)
1101 LogRel(("fHMForced=true - 64-bit guest\n"));
1102# ifdef RT_OS_DARWIN
1103 LogRel(("fHMForced=true - Darwin host\n"));
1104# endif
1105 }
1106#else /* !VBOX_WITH_RAW_MODE */
1107 fHMEnabled = fHMForced = TRUE;
1108 LogRel(("fHMForced=true - No raw-mode support in this build!\n"));
1109#endif /* !VBOX_WITH_RAW_MODE */
1110 if (!fHMForced) /* No need to query if already forced above. */
1111 {
1112 hrc = pMachine->GetHWVirtExProperty(HWVirtExPropertyType_Force, &fHMForced); H();
1113 if (fHMForced)
1114 LogRel(("fHMForced=true - HWVirtExPropertyType_Force\n"));
1115 }
1116 InsertConfigInteger(pRoot, "HMEnabled", fHMEnabled);
1117
1118 /* /EM/xzy */
1119 PCFGMNODE pEM;
1120 InsertConfigNode(pRoot, "EM", &pEM);
1121
1122 /* Triple fault behavior. */
1123 BOOL fTripleFaultReset = false;
1124 hrc = pMachine->GetCPUProperty(CPUPropertyType_TripleFaultReset, &fTripleFaultReset); H();
1125 InsertConfigInteger(pEM, "TripleFaultReset", fTripleFaultReset);
1126
1127 /* /HM/xzy */
1128 PCFGMNODE pHM;
1129 InsertConfigNode(pRoot, "HM", &pHM);
1130 InsertConfigInteger(pHM, "HMForced", fHMForced);
1131 if (fHMEnabled)
1132 {
1133 /* Indicate whether 64-bit guests are supported or not. */
1134 InsertConfigInteger(pHM, "64bitEnabled", fIsGuest64Bit);
1135#if ARCH_BITS == 32 /* The recompiler must use VBoxREM64 (32-bit host only). */
1136 PCFGMNODE pREM;
1137 InsertConfigNode(pRoot, "REM", &pREM);
1138 InsertConfigInteger(pREM, "64bitEnabled", 1);
1139#endif
1140
1141 /** @todo Not exactly pretty to check strings; VBOXOSTYPE would be better,
1142 but that requires quite a bit of API change in Main. */
1143 if ( fIOAPIC
1144 && ( osTypeId == "WindowsNT4"
1145 || osTypeId == "Windows2000"
1146 || osTypeId == "WindowsXP"
1147 || osTypeId == "Windows2003"))
1148 {
1149 /* Only allow TPR patching for NT, Win2k, XP and Windows Server 2003. (32 bits mode)
1150 * We may want to consider adding more guest OSes (Solaris) later on.
1151 */
1152 InsertConfigInteger(pHM, "TPRPatchingEnabled", 1);
1153 }
1154 }
1155
1156 /* HWVirtEx exclusive mode */
1157 BOOL fHMExclusive = true;
1158 hrc = systemProperties->COMGETTER(ExclusiveHwVirt)(&fHMExclusive); H();
1159 InsertConfigInteger(pHM, "Exclusive", fHMExclusive);
1160
1161 /* Nested paging (VT-x/AMD-V) */
1162 BOOL fEnableNestedPaging = false;
1163 hrc = pMachine->GetHWVirtExProperty(HWVirtExPropertyType_NestedPaging, &fEnableNestedPaging); H();
1164 InsertConfigInteger(pHM, "EnableNestedPaging", fEnableNestedPaging);
1165
1166 /* Large pages; requires nested paging */
1167 BOOL fEnableLargePages = false;
1168 hrc = pMachine->GetHWVirtExProperty(HWVirtExPropertyType_LargePages, &fEnableLargePages); H();
1169 InsertConfigInteger(pHM, "EnableLargePages", fEnableLargePages);
1170
1171 /* VPID (VT-x) */
1172 BOOL fEnableVPID = false;
1173 hrc = pMachine->GetHWVirtExProperty(HWVirtExPropertyType_VPID, &fEnableVPID); H();
1174 InsertConfigInteger(pHM, "EnableVPID", fEnableVPID);
1175
1176 /* Unrestricted execution aka UX (VT-x) */
1177 BOOL fEnableUX = false;
1178 hrc = pMachine->GetHWVirtExProperty(HWVirtExPropertyType_UnrestrictedExecution, &fEnableUX); H();
1179 InsertConfigInteger(pHM, "EnableUX", fEnableUX);
1180
1181 /* Reset overwrite. */
1182 if (i_isResetTurnedIntoPowerOff())
1183 InsertConfigInteger(pRoot, "PowerOffInsteadOfReset", 1);
1184
1185 /*
1186 * Paravirt. provider.
1187 */
1188 PCFGMNODE pParavirtNode;
1189 InsertConfigNode(pRoot, "GIM", &pParavirtNode);
1190 const char *pcszParavirtProvider;
1191 bool fGimDeviceNeeded = true;
1192 switch (paravirtProvider)
1193 {
1194 case ParavirtProvider_None:
1195 pcszParavirtProvider = "None";
1196 fGimDeviceNeeded = false;
1197 break;
1198
1199 case ParavirtProvider_Minimal:
1200 pcszParavirtProvider = "Minimal";
1201 break;
1202
1203 case ParavirtProvider_HyperV:
1204 pcszParavirtProvider = "HyperV";
1205 break;
1206
1207 default:
1208 AssertMsgFailed(("Invalid paravirtProvider=%d\n", paravirtProvider));
1209 return VMR3SetError(pUVM, VERR_INVALID_PARAMETER, RT_SRC_POS, N_("Invalid paravirt. provider '%d'"),
1210 paravirtProvider);
1211 }
1212 InsertConfigString(pParavirtNode, "Provider", pcszParavirtProvider);
1213
1214 /*
1215 * MM values.
1216 */
1217 PCFGMNODE pMM;
1218 InsertConfigNode(pRoot, "MM", &pMM);
1219 InsertConfigInteger(pMM, "CanUseLargerHeap", chipsetType == ChipsetType_ICH9);
1220
1221 /*
1222 * PDM config.
1223 * Load drivers in VBoxC.[so|dll]
1224 */
1225 PCFGMNODE pPDM;
1226 PCFGMNODE pNode;
1227 PCFGMNODE pMod;
1228 InsertConfigNode(pRoot, "PDM", &pPDM);
1229 InsertConfigNode(pPDM, "Devices", &pNode);
1230 InsertConfigNode(pPDM, "Drivers", &pNode);
1231 InsertConfigNode(pNode, "VBoxC", &pMod);
1232#ifdef VBOX_WITH_XPCOM
1233 // VBoxC is located in the components subdirectory
1234 char szPathVBoxC[RTPATH_MAX];
1235 rc = RTPathAppPrivateArch(szPathVBoxC, RTPATH_MAX - sizeof("/components/VBoxC")); AssertRC(rc);
1236 strcat(szPathVBoxC, "/components/VBoxC");
1237 InsertConfigString(pMod, "Path", szPathVBoxC);
1238#else
1239 InsertConfigString(pMod, "Path", "VBoxC");
1240#endif
1241
1242
1243 /*
1244 * Block cache settings.
1245 */
1246 PCFGMNODE pPDMBlkCache;
1247 InsertConfigNode(pPDM, "BlkCache", &pPDMBlkCache);
1248
1249 /* I/O cache size */
1250 ULONG ioCacheSize = 5;
1251 hrc = pMachine->COMGETTER(IOCacheSize)(&ioCacheSize); H();
1252 InsertConfigInteger(pPDMBlkCache, "CacheSize", ioCacheSize * _1M);
1253
1254 /*
1255 * Bandwidth groups.
1256 */
1257 PCFGMNODE pAc;
1258 PCFGMNODE pAcFile;
1259 PCFGMNODE pAcFileBwGroups;
1260 ComPtr<IBandwidthControl> bwCtrl;
1261 com::SafeIfaceArray<IBandwidthGroup> bwGroups;
1262
1263 hrc = pMachine->COMGETTER(BandwidthControl)(bwCtrl.asOutParam()); H();
1264
1265 hrc = bwCtrl->GetAllBandwidthGroups(ComSafeArrayAsOutParam(bwGroups)); H();
1266
1267 InsertConfigNode(pPDM, "AsyncCompletion", &pAc);
1268 InsertConfigNode(pAc, "File", &pAcFile);
1269 InsertConfigNode(pAcFile, "BwGroups", &pAcFileBwGroups);
1270#ifdef VBOX_WITH_NETSHAPER
1271 PCFGMNODE pNetworkShaper;
1272 PCFGMNODE pNetworkBwGroups;
1273
1274 InsertConfigNode(pPDM, "NetworkShaper", &pNetworkShaper);
1275 InsertConfigNode(pNetworkShaper, "BwGroups", &pNetworkBwGroups);
1276#endif /* VBOX_WITH_NETSHAPER */
1277
1278 for (size_t i = 0; i < bwGroups.size(); i++)
1279 {
1280 Bstr strName;
1281 LONG64 cMaxBytesPerSec;
1282 BandwidthGroupType_T enmType;
1283
1284 hrc = bwGroups[i]->COMGETTER(Name)(strName.asOutParam()); H();
1285 hrc = bwGroups[i]->COMGETTER(Type)(&enmType); H();
1286 hrc = bwGroups[i]->COMGETTER(MaxBytesPerSec)(&cMaxBytesPerSec); H();
1287
1288 if (strName.isEmpty())
1289 return VMR3SetError(pUVM, VERR_CFGM_NO_NODE, RT_SRC_POS,
1290 N_("No bandwidth group name specified"));
1291
1292 if (enmType == BandwidthGroupType_Disk)
1293 {
1294 PCFGMNODE pBwGroup;
1295 InsertConfigNode(pAcFileBwGroups, Utf8Str(strName).c_str(), &pBwGroup);
1296 InsertConfigInteger(pBwGroup, "Max", cMaxBytesPerSec);
1297 InsertConfigInteger(pBwGroup, "Start", cMaxBytesPerSec);
1298 InsertConfigInteger(pBwGroup, "Step", 0);
1299 }
1300#ifdef VBOX_WITH_NETSHAPER
1301 else if (enmType == BandwidthGroupType_Network)
1302 {
1303 /* Network bandwidth groups. */
1304 PCFGMNODE pBwGroup;
1305 InsertConfigNode(pNetworkBwGroups, Utf8Str(strName).c_str(), &pBwGroup);
1306 InsertConfigInteger(pBwGroup, "Max", cMaxBytesPerSec);
1307 }
1308#endif /* VBOX_WITH_NETSHAPER */
1309 }
1310
1311 /*
1312 * Devices
1313 */
1314 PCFGMNODE pDevices = NULL; /* /Devices */
1315 PCFGMNODE pDev = NULL; /* /Devices/Dev/ */
1316 PCFGMNODE pInst = NULL; /* /Devices/Dev/0/ */
1317 PCFGMNODE pCfg = NULL; /* /Devices/Dev/.../Config/ */
1318 PCFGMNODE pLunL0 = NULL; /* /Devices/Dev/0/LUN#0/ */
1319 PCFGMNODE pLunL1 = NULL; /* /Devices/Dev/0/LUN#0/AttachedDriver/ */
1320 PCFGMNODE pLunL2 = NULL; /* /Devices/Dev/0/LUN#0/AttachedDriver/Config/ */
1321 PCFGMNODE pBiosCfg = NULL; /* /Devices/pcbios/0/Config/ */
1322 PCFGMNODE pNetBootCfg = NULL; /* /Devices/pcbios/0/Config/NetBoot/ */
1323
1324 InsertConfigNode(pRoot, "Devices", &pDevices);
1325
1326 /*
1327 * GIM Device
1328 */
1329 if (fGimDeviceNeeded)
1330 {
1331 InsertConfigNode(pDevices, "GIMDev", &pDev);
1332 InsertConfigNode(pDev, "0", &pInst);
1333 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1334 //InsertConfigNode(pInst, "Config", &pCfg);
1335 }
1336
1337 /*
1338 * PC Arch.
1339 */
1340 InsertConfigNode(pDevices, "pcarch", &pDev);
1341 InsertConfigNode(pDev, "0", &pInst);
1342 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1343 InsertConfigNode(pInst, "Config", &pCfg);
1344
1345 /*
1346 * The time offset
1347 */
1348 LONG64 timeOffset;
1349 hrc = biosSettings->COMGETTER(TimeOffset)(&timeOffset); H();
1350 PCFGMNODE pTMNode;
1351 InsertConfigNode(pRoot, "TM", &pTMNode);
1352 InsertConfigInteger(pTMNode, "UTCOffset", timeOffset * 1000000);
1353
1354 /*
1355 * DMA
1356 */
1357 InsertConfigNode(pDevices, "8237A", &pDev);
1358 InsertConfigNode(pDev, "0", &pInst);
1359 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1360
1361 /*
1362 * PCI buses.
1363 */
1364 uint32_t uIocPCIAddress, uHbcPCIAddress;
1365 switch (chipsetType)
1366 {
1367 default:
1368 Assert(false);
1369 case ChipsetType_PIIX3:
1370 InsertConfigNode(pDevices, "pci", &pDev);
1371 uHbcPCIAddress = (0x0 << 16) | 0;
1372 uIocPCIAddress = (0x1 << 16) | 0; // ISA controller
1373 break;
1374 case ChipsetType_ICH9:
1375 InsertConfigNode(pDevices, "ich9pci", &pDev);
1376 uHbcPCIAddress = (0x1e << 16) | 0;
1377 uIocPCIAddress = (0x1f << 16) | 0; // LPC controller
1378 break;
1379 }
1380 InsertConfigNode(pDev, "0", &pInst);
1381 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1382 InsertConfigNode(pInst, "Config", &pCfg);
1383 InsertConfigInteger(pCfg, "IOAPIC", fIOAPIC);
1384 if (chipsetType == ChipsetType_ICH9)
1385 {
1386 /* Provide MCFG info */
1387 InsertConfigInteger(pCfg, "McfgBase", uMcfgBase);
1388 InsertConfigInteger(pCfg, "McfgLength", cbMcfgLength);
1389
1390
1391 /* And register 2 bridges */
1392 InsertConfigNode(pDevices, "ich9pcibridge", &pDev);
1393 InsertConfigNode(pDev, "0", &pInst);
1394 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1395 hrc = pBusMgr->assignPCIDevice("ich9pcibridge", pInst); H();
1396
1397 InsertConfigNode(pDev, "1", &pInst);
1398 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1399 hrc = pBusMgr->assignPCIDevice("ich9pcibridge", pInst); H();
1400
1401#ifdef VBOX_WITH_PCI_PASSTHROUGH
1402 /* Add PCI passthrough devices */
1403 hrc = i_attachRawPCIDevices(pUVM, pBusMgr, pDevices); H();
1404#endif
1405 }
1406
1407 /*
1408 * Enable the following devices: HPET, SMC and LPC on MacOS X guests or on ICH9 chipset
1409 */
1410
1411 /*
1412 * High Precision Event Timer (HPET)
1413 */
1414 BOOL fHPETEnabled;
1415 /* Other guests may wish to use HPET too, but MacOS X not functional without it */
1416 hrc = pMachine->COMGETTER(HPETEnabled)(&fHPETEnabled); H();
1417 /* so always enable HPET in extended profile */
1418 fHPETEnabled |= fOsXGuest;
1419 /* HPET is always present on ICH9 */
1420 fHPETEnabled |= (chipsetType == ChipsetType_ICH9);
1421 if (fHPETEnabled)
1422 {
1423 InsertConfigNode(pDevices, "hpet", &pDev);
1424 InsertConfigNode(pDev, "0", &pInst);
1425 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1426 InsertConfigNode(pInst, "Config", &pCfg);
1427 InsertConfigInteger(pCfg, "ICH9", (chipsetType == ChipsetType_ICH9) ? 1 : 0); /* boolean */
1428 }
1429
1430 /*
1431 * System Management Controller (SMC)
1432 */
1433 BOOL fSmcEnabled;
1434 fSmcEnabled = fOsXGuest;
1435 if (fSmcEnabled)
1436 {
1437 InsertConfigNode(pDevices, "smc", &pDev);
1438 InsertConfigNode(pDev, "0", &pInst);
1439 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1440 InsertConfigNode(pInst, "Config", &pCfg);
1441
1442 bool fGetKeyFromRealSMC;
1443 Utf8Str strKey;
1444 rc = getSmcDeviceKey(virtualBox, pMachine, &strKey, &fGetKeyFromRealSMC);
1445 AssertRCReturn(rc, rc);
1446
1447 InsertConfigString(pCfg, "DeviceKey", strKey);
1448 InsertConfigInteger(pCfg, "GetKeyFromRealSMC", fGetKeyFromRealSMC);
1449 }
1450
1451 /*
1452 * Low Pin Count (LPC) bus
1453 */
1454 BOOL fLpcEnabled;
1455 /** @todo: implement appropriate getter */
1456 fLpcEnabled = fOsXGuest || (chipsetType == ChipsetType_ICH9);
1457 if (fLpcEnabled)
1458 {
1459 InsertConfigNode(pDevices, "lpc", &pDev);
1460 InsertConfigNode(pDev, "0", &pInst);
1461 hrc = pBusMgr->assignPCIDevice("lpc", pInst); H();
1462 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1463 }
1464
1465 BOOL fShowRtc;
1466 fShowRtc = fOsXGuest || (chipsetType == ChipsetType_ICH9);
1467
1468 /*
1469 * PS/2 keyboard & mouse.
1470 */
1471 InsertConfigNode(pDevices, "pckbd", &pDev);
1472 InsertConfigNode(pDev, "0", &pInst);
1473 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1474 InsertConfigNode(pInst, "Config", &pCfg);
1475
1476 InsertConfigNode(pInst, "LUN#0", &pLunL0);
1477 InsertConfigString(pLunL0, "Driver", "KeyboardQueue");
1478 InsertConfigNode(pLunL0, "Config", &pCfg);
1479 InsertConfigInteger(pCfg, "QueueSize", 64);
1480
1481 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
1482 InsertConfigString(pLunL1, "Driver", "MainKeyboard");
1483 InsertConfigNode(pLunL1, "Config", &pCfg);
1484 Keyboard *pKeyboard = mKeyboard;
1485 InsertConfigInteger(pCfg, "Object", (uintptr_t)pKeyboard);
1486
1487 Mouse *pMouse = mMouse;
1488 PointingHIDType_T aPointingHID;
1489 hrc = pMachine->COMGETTER(PointingHIDType)(&aPointingHID); H();
1490 InsertConfigNode(pInst, "LUN#1", &pLunL0);
1491 InsertConfigString(pLunL0, "Driver", "MouseQueue");
1492 InsertConfigNode(pLunL0, "Config", &pCfg);
1493 InsertConfigInteger(pCfg, "QueueSize", 128);
1494
1495 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
1496 InsertConfigString(pLunL1, "Driver", "MainMouse");
1497 InsertConfigNode(pLunL1, "Config", &pCfg);
1498 InsertConfigInteger(pCfg, "Object", (uintptr_t)pMouse);
1499
1500 /*
1501 * i8254 Programmable Interval Timer And Dummy Speaker
1502 */
1503 InsertConfigNode(pDevices, "i8254", &pDev);
1504 InsertConfigNode(pDev, "0", &pInst);
1505 InsertConfigNode(pInst, "Config", &pCfg);
1506#ifdef DEBUG
1507 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1508#endif
1509
1510 /*
1511 * i8259 Programmable Interrupt Controller.
1512 */
1513 InsertConfigNode(pDevices, "i8259", &pDev);
1514 InsertConfigNode(pDev, "0", &pInst);
1515 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1516 InsertConfigNode(pInst, "Config", &pCfg);
1517
1518 /*
1519 * Advanced Programmable Interrupt Controller.
1520 * SMP: Each CPU has a LAPIC, but we have a single device representing all LAPICs states,
1521 * thus only single insert
1522 */
1523 InsertConfigNode(pDevices, "apic", &pDev);
1524 InsertConfigNode(pDev, "0", &pInst);
1525 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1526 InsertConfigNode(pInst, "Config", &pCfg);
1527 InsertConfigInteger(pCfg, "IOAPIC", fIOAPIC);
1528 InsertConfigInteger(pCfg, "NumCPUs", cCpus);
1529
1530 if (fIOAPIC)
1531 {
1532 /*
1533 * I/O Advanced Programmable Interrupt Controller.
1534 */
1535 InsertConfigNode(pDevices, "ioapic", &pDev);
1536 InsertConfigNode(pDev, "0", &pInst);
1537 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1538 InsertConfigNode(pInst, "Config", &pCfg);
1539 InsertConfigInteger(pCfg, "NumCPUs", cCpus);
1540 }
1541
1542 /*
1543 * RTC MC146818.
1544 */
1545 InsertConfigNode(pDevices, "mc146818", &pDev);
1546 InsertConfigNode(pDev, "0", &pInst);
1547 InsertConfigNode(pInst, "Config", &pCfg);
1548 BOOL fRTCUseUTC;
1549 hrc = pMachine->COMGETTER(RTCUseUTC)(&fRTCUseUTC); H();
1550 InsertConfigInteger(pCfg, "UseUTC", fRTCUseUTC ? 1 : 0);
1551
1552 /*
1553 * VGA.
1554 */
1555 GraphicsControllerType_T enmGraphicsController;
1556 hrc = pMachine->COMGETTER(GraphicsControllerType)(&enmGraphicsController); H();
1557 switch (enmGraphicsController)
1558 {
1559 case GraphicsControllerType_Null:
1560 break;
1561 case GraphicsControllerType_VBoxVGA:
1562#ifdef VBOX_WITH_VMSVGA
1563 case GraphicsControllerType_VMSVGA:
1564#endif
1565 rc = i_configGraphicsController(pDevices, enmGraphicsController, pBusMgr, pMachine, biosSettings,
1566 RT_BOOL(fHMEnabled));
1567 if (FAILED(rc))
1568 return rc;
1569 break;
1570 default:
1571 AssertMsgFailed(("Invalid graphicsController=%d\n", enmGraphicsController));
1572 return VMR3SetError(pUVM, VERR_INVALID_PARAMETER, RT_SRC_POS,
1573 N_("Invalid graphics controller type '%d'"), enmGraphicsController);
1574 }
1575
1576 /*
1577 * Firmware.
1578 */
1579 FirmwareType_T eFwType = FirmwareType_BIOS;
1580 hrc = pMachine->COMGETTER(FirmwareType)(&eFwType); H();
1581
1582#ifdef VBOX_WITH_EFI
1583 BOOL fEfiEnabled = (eFwType >= FirmwareType_EFI) && (eFwType <= FirmwareType_EFIDUAL);
1584#else
1585 BOOL fEfiEnabled = false;
1586#endif
1587 if (!fEfiEnabled)
1588 {
1589 /*
1590 * PC Bios.
1591 */
1592 InsertConfigNode(pDevices, "pcbios", &pDev);
1593 InsertConfigNode(pDev, "0", &pInst);
1594 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1595 InsertConfigNode(pInst, "Config", &pBiosCfg);
1596 InsertConfigInteger(pBiosCfg, "RamSize", cbRam);
1597 InsertConfigInteger(pBiosCfg, "RamHoleSize", cbRamHole);
1598 InsertConfigInteger(pBiosCfg, "NumCPUs", cCpus);
1599 InsertConfigString(pBiosCfg, "HardDiskDevice", "piix3ide");
1600 InsertConfigString(pBiosCfg, "FloppyDevice", "i82078");
1601 InsertConfigInteger(pBiosCfg, "IOAPIC", fIOAPIC);
1602 BOOL fPXEDebug;
1603 hrc = biosSettings->COMGETTER(PXEDebugEnabled)(&fPXEDebug); H();
1604 InsertConfigInteger(pBiosCfg, "PXEDebug", fPXEDebug);
1605 InsertConfigBytes(pBiosCfg, "UUID", &HardwareUuid,sizeof(HardwareUuid));
1606 InsertConfigNode(pBiosCfg, "NetBoot", &pNetBootCfg);
1607 InsertConfigInteger(pBiosCfg, "McfgBase", uMcfgBase);
1608 InsertConfigInteger(pBiosCfg, "McfgLength", cbMcfgLength);
1609
1610 DeviceType_T bootDevice;
1611 AssertMsgReturn(SchemaDefs::MaxBootPosition <= 9, ("Too many boot devices %d\n", SchemaDefs::MaxBootPosition),
1612 VERR_INVALID_PARAMETER);
1613
1614 for (ULONG pos = 1; pos <= SchemaDefs::MaxBootPosition; ++pos)
1615 {
1616 hrc = pMachine->GetBootOrder(pos, &bootDevice); H();
1617
1618 char szParamName[] = "BootDeviceX";
1619 szParamName[sizeof(szParamName) - 2] = ((char (pos - 1)) + '0');
1620
1621 const char *pszBootDevice;
1622 switch (bootDevice)
1623 {
1624 case DeviceType_Null:
1625 pszBootDevice = "NONE";
1626 break;
1627 case DeviceType_HardDisk:
1628 pszBootDevice = "IDE";
1629 break;
1630 case DeviceType_DVD:
1631 pszBootDevice = "DVD";
1632 break;
1633 case DeviceType_Floppy:
1634 pszBootDevice = "FLOPPY";
1635 break;
1636 case DeviceType_Network:
1637 pszBootDevice = "LAN";
1638 break;
1639 default:
1640 AssertMsgFailed(("Invalid bootDevice=%d\n", bootDevice));
1641 return VMR3SetError(pUVM, VERR_INVALID_PARAMETER, RT_SRC_POS,
1642 N_("Invalid boot device '%d'"), bootDevice);
1643 }
1644 InsertConfigString(pBiosCfg, szParamName, pszBootDevice);
1645 }
1646
1647 /** @todo @bugref{7145}: We might want to enable this by default for new VMs. For now,
1648 * this is required for Windows 2012 guests. */
1649 if (osTypeId == "Windows2012_64")
1650 InsertConfigInteger(pBiosCfg, "DmiExposeMemoryTable", 1); /* boolean */
1651 }
1652 else
1653 {
1654 /* Autodetect firmware type, basing on guest type */
1655 if (eFwType == FirmwareType_EFI)
1656 {
1657 eFwType = fIsGuest64Bit
1658 ? (FirmwareType_T)FirmwareType_EFI64
1659 : (FirmwareType_T)FirmwareType_EFI32;
1660 }
1661 bool const f64BitEntry = eFwType == FirmwareType_EFI64;
1662
1663 Utf8Str efiRomFile;
1664 rc = findEfiRom(virtualBox, eFwType, &efiRomFile);
1665 AssertRCReturn(rc, rc);
1666
1667 /* Get boot args */
1668 Utf8Str bootArgs;
1669 GetExtraDataBoth(virtualBox, pMachine, "VBoxInternal2/EfiBootArgs", &bootArgs);
1670
1671 /* Get device props */
1672 Utf8Str deviceProps;
1673 GetExtraDataBoth(virtualBox, pMachine, "VBoxInternal2/EfiDeviceProps", &deviceProps);
1674
1675 /* Get GOP mode settings */
1676 uint32_t u32GopMode = UINT32_MAX;
1677 GetExtraDataBoth(virtualBox, pMachine, "VBoxInternal2/EfiGopMode", &strTmp);
1678 if (!strTmp.isEmpty())
1679 u32GopMode = strTmp.toUInt32();
1680
1681 /* UGA mode settings */
1682 uint32_t u32UgaHorizontal = 0;
1683 GetExtraDataBoth(virtualBox, pMachine, "VBoxInternal2/EfiUgaHorizontalResolution", &strTmp);
1684 if (!strTmp.isEmpty())
1685 u32UgaHorizontal = strTmp.toUInt32();
1686
1687 uint32_t u32UgaVertical = 0;
1688 GetExtraDataBoth(virtualBox, pMachine, "VBoxInternal2/EfiUgaVerticalResolution", &strTmp);
1689 if (!strTmp.isEmpty())
1690 u32UgaVertical = strTmp.toUInt32();
1691
1692 /*
1693 * EFI subtree.
1694 */
1695 InsertConfigNode(pDevices, "efi", &pDev);
1696 InsertConfigNode(pDev, "0", &pInst);
1697 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1698 InsertConfigNode(pInst, "Config", &pCfg);
1699 InsertConfigInteger(pCfg, "RamSize", cbRam);
1700 InsertConfigInteger(pCfg, "RamHoleSize", cbRamHole);
1701 InsertConfigInteger(pCfg, "NumCPUs", cCpus);
1702 InsertConfigString(pCfg, "EfiRom", efiRomFile);
1703 InsertConfigString(pCfg, "BootArgs", bootArgs);
1704 InsertConfigString(pCfg, "DeviceProps", deviceProps);
1705 InsertConfigInteger(pCfg, "IOAPIC", fIOAPIC);
1706 InsertConfigBytes(pCfg, "UUID", &HardwareUuid,sizeof(HardwareUuid));
1707 InsertConfigInteger(pCfg, "64BitEntry", f64BitEntry); /* boolean */
1708 InsertConfigInteger(pCfg, "GopMode", u32GopMode);
1709 InsertConfigInteger(pCfg, "UgaHorizontalResolution", u32UgaHorizontal);
1710 InsertConfigInteger(pCfg, "UgaVerticalResolution", u32UgaVertical);
1711
1712 /* For OS X guests we'll force passing host's DMI info to the guest */
1713 if (fOsXGuest)
1714 {
1715 InsertConfigInteger(pCfg, "DmiUseHostInfo", 1);
1716 InsertConfigInteger(pCfg, "DmiExposeMemoryTable", 1);
1717 }
1718 InsertConfigNode(pInst, "LUN#0", &pLunL0);
1719 InsertConfigString(pLunL0, "Driver", "NvramStorage");
1720 InsertConfigNode(pLunL0, "Config", &pCfg);
1721 InsertConfigInteger(pCfg, "Object", (uintptr_t)mNvram);
1722#ifdef DEBUG_vvl
1723 InsertConfigInteger(pCfg, "PermanentSave", 1);
1724#endif
1725 }
1726
1727 /*
1728 * The USB Controllers.
1729 */
1730 com::SafeIfaceArray<IUSBController> usbCtrls;
1731 hrc = pMachine->COMGETTER(USBControllers)(ComSafeArrayAsOutParam(usbCtrls)); H();
1732 bool fOhciPresent = false; /**< Flag whether at least one OHCI controller is present. */
1733 bool fXhciPresent = false; /**< Flag whether at least one XHCI controller is present. */
1734
1735 for (size_t i = 0; i < usbCtrls.size(); ++i)
1736 {
1737 USBControllerType_T enmCtrlType;
1738 rc = usbCtrls[i]->COMGETTER(Type)(&enmCtrlType); H();
1739 if (enmCtrlType == USBControllerType_OHCI)
1740 {
1741 fOhciPresent = true;
1742 break;
1743 }
1744 else if (enmCtrlType == USBControllerType_XHCI)
1745 {
1746 fXhciPresent = true;
1747 break;
1748 }
1749 }
1750
1751 /*
1752 * Currently EHCI is only enabled when an OHCI or XHCI controller is present as well.
1753 */
1754 if (fOhciPresent || fXhciPresent)
1755 mfVMHasUsbController = true;
1756
1757 PCFGMNODE pUsbDevices = NULL; /**< Required for USB storage controller later. */
1758 if (mfVMHasUsbController)
1759 {
1760 for (size_t i = 0; i < usbCtrls.size(); ++i)
1761 {
1762 USBControllerType_T enmCtrlType;
1763 rc = usbCtrls[i]->COMGETTER(Type)(&enmCtrlType); H();
1764
1765 if (enmCtrlType == USBControllerType_OHCI)
1766 {
1767 InsertConfigNode(pDevices, "usb-ohci", &pDev);
1768 InsertConfigNode(pDev, "0", &pInst);
1769 InsertConfigNode(pInst, "Config", &pCfg);
1770 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1771 hrc = pBusMgr->assignPCIDevice("usb-ohci", pInst); H();
1772 InsertConfigNode(pInst, "LUN#0", &pLunL0);
1773 InsertConfigString(pLunL0, "Driver", "VUSBRootHub");
1774 InsertConfigNode(pLunL0, "Config", &pCfg);
1775
1776 /*
1777 * Attach the status driver.
1778 */
1779 i_attachStatusDriver(pInst, &mapUSBLed[0], 0, 0, NULL, NULL, 0);
1780 }
1781#ifdef VBOX_WITH_EHCI
1782 else if (enmCtrlType == USBControllerType_EHCI)
1783 {
1784 /*
1785 * USB 2.0 is only available if the proper ExtPack is installed.
1786 *
1787 * Note. Configuring EHCI here and providing messages about
1788 * the missing extpack isn't exactly clean, but it is a
1789 * necessary evil to patch over legacy compatability issues
1790 * introduced by the new distribution model.
1791 */
1792 static const char *s_pszUsbExtPackName = "Oracle VM VirtualBox Extension Pack";
1793# ifdef VBOX_WITH_EXTPACK
1794 if (mptrExtPackManager->i_isExtPackUsable(s_pszUsbExtPackName))
1795# endif
1796 {
1797 InsertConfigNode(pDevices, "usb-ehci", &pDev);
1798 InsertConfigNode(pDev, "0", &pInst);
1799 InsertConfigNode(pInst, "Config", &pCfg);
1800 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1801 hrc = pBusMgr->assignPCIDevice("usb-ehci", pInst); H();
1802
1803 InsertConfigNode(pInst, "LUN#0", &pLunL0);
1804 InsertConfigString(pLunL0, "Driver", "VUSBRootHub");
1805 InsertConfigNode(pLunL0, "Config", &pCfg);
1806
1807 /*
1808 * Attach the status driver.
1809 */
1810 i_attachStatusDriver(pInst, &mapUSBLed[1], 0, 0, NULL, NULL, 0);
1811 }
1812# ifdef VBOX_WITH_EXTPACK
1813 else
1814 {
1815 /* Always fatal! Up to VBox 4.0.4 we allowed to start the VM anyway
1816 * but this induced problems when the user saved + restored the VM! */
1817 return VMR3SetError(pUVM, VERR_NOT_FOUND, RT_SRC_POS,
1818 N_("Implementation of the USB 2.0 controller not found!\n"
1819 "Because the USB 2.0 controller state is part of the saved "
1820 "VM state, the VM cannot be started. To fix "
1821 "this problem, either install the '%s' or disable USB 2.0 "
1822 "support in the VM settings"),
1823 s_pszUsbExtPackName);
1824 }
1825# endif
1826 }
1827#endif
1828 else if (enmCtrlType == USBControllerType_XHCI)
1829 {
1830 /*
1831 * USB 3.0 is only available if the proper ExtPack is installed.
1832 *
1833 * Note. Configuring EHCI here and providing messages about
1834 * the missing extpack isn't exactly clean, but it is a
1835 * necessary evil to patch over legacy compatability issues
1836 * introduced by the new distribution model.
1837 */
1838 static const char *s_pszUsbExtPackName = "Oracle VM VirtualBox Extension Pack";
1839# ifdef VBOX_WITH_EXTPACK
1840 if (mptrExtPackManager->i_isExtPackUsable(s_pszUsbExtPackName))
1841# endif
1842 {
1843 InsertConfigNode(pDevices, "usb-xhci", &pDev);
1844 InsertConfigNode(pDev, "0", &pInst);
1845 InsertConfigNode(pInst, "Config", &pCfg);
1846 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
1847 hrc = pBusMgr->assignPCIDevice("usb-xhci", pInst); H();
1848
1849 InsertConfigNode(pInst, "LUN#0", &pLunL0);
1850 InsertConfigString(pLunL0, "Driver", "VUSBRootHub");
1851 InsertConfigNode(pLunL0, "Config", &pCfg);
1852
1853 InsertConfigNode(pInst, "LUN#1", &pLunL1);
1854 InsertConfigString(pLunL1, "Driver", "VUSBRootHub");
1855 InsertConfigNode(pLunL1, "Config", &pCfg);
1856
1857 /*
1858 * Attach the status driver.
1859 */
1860 i_attachStatusDriver(pInst, &mapUSBLed[0], 0, 1, NULL, NULL, 0);
1861 }
1862# ifdef VBOX_WITH_EXTPACK
1863 else
1864 {
1865 /* Always fatal. */
1866 return VMR3SetError(pUVM, VERR_NOT_FOUND, RT_SRC_POS,
1867 N_("Implementation of the USB 3.0 controller not found!\n"
1868 "Because the USB 3.0 controller state is part of the saved "
1869 "VM state, the VM cannot be started. To fix "
1870 "this problem, either install the '%s' or disable USB 3.0 "
1871 "support in the VM settings"),
1872 s_pszUsbExtPackName);
1873 }
1874# endif
1875 }
1876 } /* for every USB controller. */
1877
1878
1879 /*
1880 * Virtual USB Devices.
1881 */
1882 InsertConfigNode(pRoot, "USB", &pUsbDevices);
1883
1884#ifdef VBOX_WITH_USB
1885 {
1886 /*
1887 * Global USB options, currently unused as we'll apply the 2.0 -> 1.1 morphing
1888 * on a per device level now.
1889 */
1890 InsertConfigNode(pUsbDevices, "USBProxy", &pCfg);
1891 InsertConfigNode(pCfg, "GlobalConfig", &pCfg);
1892 // This globally enables the 2.0 -> 1.1 device morphing of proxied devices to keep windows quiet.
1893 //InsertConfigInteger(pCfg, "Force11Device", true);
1894 // The following breaks stuff, but it makes MSDs work in vista. (I include it here so
1895 // that it's documented somewhere.) Users needing it can use:
1896 // VBoxManage setextradata "myvm" "VBoxInternal/USB/USBProxy/GlobalConfig/Force11PacketSize" 1
1897 //InsertConfigInteger(pCfg, "Force11PacketSize", true);
1898 }
1899#endif
1900
1901#ifdef VBOX_WITH_USB_CARDREADER
1902 BOOL aEmulatedUSBCardReaderEnabled = FALSE;
1903 hrc = pMachine->COMGETTER(EmulatedUSBCardReaderEnabled)(&aEmulatedUSBCardReaderEnabled); H();
1904 if (aEmulatedUSBCardReaderEnabled)
1905 {
1906 InsertConfigNode(pUsbDevices, "CardReader", &pDev);
1907 InsertConfigNode(pDev, "0", &pInst);
1908 InsertConfigNode(pInst, "Config", &pCfg);
1909
1910 InsertConfigNode(pInst, "LUN#0", &pLunL0);
1911# ifdef VBOX_WITH_USB_CARDREADER_TEST
1912 InsertConfigString(pLunL0, "Driver", "DrvDirectCardReader");
1913 InsertConfigNode(pLunL0, "Config", &pCfg);
1914# else
1915 InsertConfigString(pLunL0, "Driver", "UsbCardReader");
1916 InsertConfigNode(pLunL0, "Config", &pCfg);
1917 InsertConfigInteger(pCfg, "Object", (uintptr_t)mUsbCardReader);
1918# endif
1919 }
1920#endif
1921
1922 /* Virtual USB Mouse/Tablet */
1923 if ( aPointingHID == PointingHIDType_USBMouse
1924 || aPointingHID == PointingHIDType_USBTablet
1925 || aPointingHID == PointingHIDType_USBMultiTouch)
1926 {
1927 InsertConfigNode(pUsbDevices, "HidMouse", &pDev);
1928 InsertConfigNode(pDev, "0", &pInst);
1929 InsertConfigNode(pInst, "Config", &pCfg);
1930
1931 if (aPointingHID == PointingHIDType_USBMouse)
1932 InsertConfigString(pCfg, "Mode", "relative");
1933 else
1934 InsertConfigString(pCfg, "Mode", "absolute");
1935 InsertConfigNode(pInst, "LUN#0", &pLunL0);
1936 InsertConfigString(pLunL0, "Driver", "MouseQueue");
1937 InsertConfigNode(pLunL0, "Config", &pCfg);
1938 InsertConfigInteger(pCfg, "QueueSize", 128);
1939
1940 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
1941 InsertConfigString(pLunL1, "Driver", "MainMouse");
1942 InsertConfigNode(pLunL1, "Config", &pCfg);
1943 InsertConfigInteger(pCfg, "Object", (uintptr_t)pMouse);
1944 }
1945 if (aPointingHID == PointingHIDType_USBMultiTouch)
1946 {
1947 InsertConfigNode(pDev, "1", &pInst);
1948 InsertConfigNode(pInst, "Config", &pCfg);
1949
1950 InsertConfigString(pCfg, "Mode", "multitouch");
1951 InsertConfigNode(pInst, "LUN#0", &pLunL0);
1952 InsertConfigString(pLunL0, "Driver", "MouseQueue");
1953 InsertConfigNode(pLunL0, "Config", &pCfg);
1954 InsertConfigInteger(pCfg, "QueueSize", 128);
1955
1956 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
1957 InsertConfigString(pLunL1, "Driver", "MainMouse");
1958 InsertConfigNode(pLunL1, "Config", &pCfg);
1959 InsertConfigInteger(pCfg, "Object", (uintptr_t)pMouse);
1960 }
1961
1962 /* Virtual USB Keyboard */
1963 KeyboardHIDType_T aKbdHID;
1964 hrc = pMachine->COMGETTER(KeyboardHIDType)(&aKbdHID); H();
1965 if (aKbdHID == KeyboardHIDType_USBKeyboard)
1966 {
1967 InsertConfigNode(pUsbDevices, "HidKeyboard", &pDev);
1968 InsertConfigNode(pDev, "0", &pInst);
1969 InsertConfigNode(pInst, "Config", &pCfg);
1970
1971 InsertConfigNode(pInst, "LUN#0", &pLunL0);
1972 InsertConfigString(pLunL0, "Driver", "KeyboardQueue");
1973 InsertConfigNode(pLunL0, "Config", &pCfg);
1974 InsertConfigInteger(pCfg, "QueueSize", 64);
1975
1976 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
1977 InsertConfigString(pLunL1, "Driver", "MainKeyboard");
1978 InsertConfigNode(pLunL1, "Config", &pCfg);
1979 pKeyboard = mKeyboard;
1980 InsertConfigInteger(pCfg, "Object", (uintptr_t)pKeyboard);
1981 }
1982 }
1983
1984 /*
1985 * Storage controllers.
1986 */
1987 com::SafeIfaceArray<IStorageController> ctrls;
1988 PCFGMNODE aCtrlNodes[StorageControllerType_LsiLogicSas + 1] = {};
1989 hrc = pMachine->COMGETTER(StorageControllers)(ComSafeArrayAsOutParam(ctrls)); H();
1990
1991 bool fFdcEnabled = false;
1992 for (size_t i = 0; i < ctrls.size(); ++i)
1993 {
1994 DeviceType_T *paLedDevType = NULL;
1995
1996 StorageControllerType_T enmCtrlType;
1997 rc = ctrls[i]->COMGETTER(ControllerType)(&enmCtrlType); H();
1998 AssertRelease((unsigned)enmCtrlType < RT_ELEMENTS(aCtrlNodes)
1999 || enmCtrlType == StorageControllerType_USB);
2000
2001 StorageBus_T enmBus;
2002 rc = ctrls[i]->COMGETTER(Bus)(&enmBus); H();
2003
2004 Bstr controllerName;
2005 rc = ctrls[i]->COMGETTER(Name)(controllerName.asOutParam()); H();
2006
2007 ULONG ulInstance = 999;
2008 rc = ctrls[i]->COMGETTER(Instance)(&ulInstance); H();
2009
2010 BOOL fUseHostIOCache;
2011 rc = ctrls[i]->COMGETTER(UseHostIOCache)(&fUseHostIOCache); H();
2012
2013 BOOL fBootable;
2014 rc = ctrls[i]->COMGETTER(Bootable)(&fBootable); H();
2015
2016 PCFGMNODE pCtlInst = NULL;
2017 const char *pszCtrlDev = i_convertControllerTypeToDev(enmCtrlType);
2018 if (enmCtrlType != StorageControllerType_USB)
2019 {
2020 /* /Devices/<ctrldev>/ */
2021 pDev = aCtrlNodes[enmCtrlType];
2022 if (!pDev)
2023 {
2024 InsertConfigNode(pDevices, pszCtrlDev, &pDev);
2025 aCtrlNodes[enmCtrlType] = pDev; /* IDE variants are handled in the switch */
2026 }
2027
2028 /* /Devices/<ctrldev>/<instance>/ */
2029 InsertConfigNode(pDev, Utf8StrFmt("%u", ulInstance).c_str(), &pCtlInst);
2030
2031 /* Device config: /Devices/<ctrldev>/<instance>/<values> & /ditto/Config/<values> */
2032 InsertConfigInteger(pCtlInst, "Trusted", 1);
2033 InsertConfigNode(pCtlInst, "Config", &pCfg);
2034 }
2035
2036 static const char * const apszBiosConfigScsi[MAX_BIOS_LUN_COUNT] =
2037 { "ScsiLUN1", "ScsiLUN2", "ScsiLUN3", "ScsiLUN4" };
2038
2039 static const char * const apszBiosConfigSata[MAX_BIOS_LUN_COUNT] =
2040 { "SataLUN1", "SataLUN2", "SataLUN3", "SataLUN4" };
2041
2042 switch (enmCtrlType)
2043 {
2044 case StorageControllerType_LsiLogic:
2045 {
2046 hrc = pBusMgr->assignPCIDevice("lsilogic", pCtlInst); H();
2047
2048 InsertConfigInteger(pCfg, "Bootable", fBootable);
2049
2050 /* BIOS configuration values, first SCSI controller only. */
2051 if ( !pBusMgr->hasPCIDevice("lsilogic", 1)
2052 && !pBusMgr->hasPCIDevice("buslogic", 0)
2053 && !pBusMgr->hasPCIDevice("lsilogicsas", 0)
2054 && pBiosCfg)
2055 {
2056 InsertConfigString(pBiosCfg, "ScsiHardDiskDevice", "lsilogicscsi");
2057 hrc = SetBiosDiskInfo(pMachine, pCfg, pBiosCfg, controllerName, apszBiosConfigScsi); H();
2058 }
2059
2060 /* Attach the status driver */
2061 Assert(cLedScsi >= 16);
2062 i_attachStatusDriver(pCtlInst, &mapStorageLeds[iLedScsi], 0, 15,
2063 &mapMediumAttachments, pszCtrlDev, ulInstance);
2064 paLedDevType = &maStorageDevType[iLedScsi];
2065 break;
2066 }
2067
2068 case StorageControllerType_BusLogic:
2069 {
2070 hrc = pBusMgr->assignPCIDevice("buslogic", pCtlInst); H();
2071
2072 InsertConfigInteger(pCfg, "Bootable", fBootable);
2073
2074 /* BIOS configuration values, first SCSI controller only. */
2075 if ( !pBusMgr->hasPCIDevice("lsilogic", 0)
2076 && !pBusMgr->hasPCIDevice("buslogic", 1)
2077 && !pBusMgr->hasPCIDevice("lsilogicsas", 0)
2078 && pBiosCfg)
2079 {
2080 InsertConfigString(pBiosCfg, "ScsiHardDiskDevice", "buslogic");
2081 hrc = SetBiosDiskInfo(pMachine, pCfg, pBiosCfg, controllerName, apszBiosConfigScsi); H();
2082 }
2083
2084 /* Attach the status driver */
2085 Assert(cLedScsi >= 16);
2086 i_attachStatusDriver(pCtlInst, &mapStorageLeds[iLedScsi], 0, 15,
2087 &mapMediumAttachments, pszCtrlDev, ulInstance);
2088 paLedDevType = &maStorageDevType[iLedScsi];
2089 break;
2090 }
2091
2092 case StorageControllerType_IntelAhci:
2093 {
2094 hrc = pBusMgr->assignPCIDevice("ahci", pCtlInst); H();
2095
2096 ULONG cPorts = 0;
2097 hrc = ctrls[i]->COMGETTER(PortCount)(&cPorts); H();
2098 InsertConfigInteger(pCfg, "PortCount", cPorts);
2099 InsertConfigInteger(pCfg, "Bootable", fBootable);
2100
2101 /* BIOS configuration values, first AHCI controller only. */
2102 if ( !pBusMgr->hasPCIDevice("ahci", 1)
2103 && pBiosCfg)
2104 {
2105 InsertConfigString(pBiosCfg, "SataHardDiskDevice", "ahci");
2106 hrc = SetBiosDiskInfo(pMachine, pCfg, pBiosCfg, controllerName, apszBiosConfigSata); H();
2107 }
2108
2109 /* Attach the status driver */
2110 AssertRelease(cPorts <= cLedSata);
2111 i_attachStatusDriver(pCtlInst, &mapStorageLeds[iLedSata], 0, cPorts - 1,
2112 &mapMediumAttachments, pszCtrlDev, ulInstance);
2113 paLedDevType = &maStorageDevType[iLedSata];
2114 break;
2115 }
2116
2117 case StorageControllerType_PIIX3:
2118 case StorageControllerType_PIIX4:
2119 case StorageControllerType_ICH6:
2120 {
2121 /*
2122 * IDE (update this when the main interface changes)
2123 */
2124 hrc = pBusMgr->assignPCIDevice("piix3ide", pCtlInst); H();
2125 InsertConfigString(pCfg, "Type", controllerString(enmCtrlType));
2126 /* Attach the status driver */
2127 Assert(cLedIde >= 4);
2128 i_attachStatusDriver(pCtlInst, &mapStorageLeds[iLedIde], 0, 3,
2129 &mapMediumAttachments, pszCtrlDev, ulInstance);
2130 paLedDevType = &maStorageDevType[iLedIde];
2131
2132 /* IDE flavors */
2133 aCtrlNodes[StorageControllerType_PIIX3] = pDev;
2134 aCtrlNodes[StorageControllerType_PIIX4] = pDev;
2135 aCtrlNodes[StorageControllerType_ICH6] = pDev;
2136 break;
2137 }
2138
2139 case StorageControllerType_I82078:
2140 {
2141 /*
2142 * i82078 Floppy drive controller
2143 */
2144 fFdcEnabled = true;
2145 InsertConfigInteger(pCfg, "IRQ", 6);
2146 InsertConfigInteger(pCfg, "DMA", 2);
2147 InsertConfigInteger(pCfg, "MemMapped", 0 );
2148 InsertConfigInteger(pCfg, "IOBase", 0x3f0);
2149
2150 /* Attach the status driver */
2151 Assert(cLedFloppy >= 2);
2152 i_attachStatusDriver(pCtlInst, &mapStorageLeds[iLedFloppy], 0, 1,
2153 &mapMediumAttachments, pszCtrlDev, ulInstance);
2154 paLedDevType = &maStorageDevType[iLedFloppy];
2155 break;
2156 }
2157
2158 case StorageControllerType_LsiLogicSas:
2159 {
2160 hrc = pBusMgr->assignPCIDevice("lsilogicsas", pCtlInst); H();
2161
2162 InsertConfigString(pCfg, "ControllerType", "SAS1068");
2163 InsertConfigInteger(pCfg, "Bootable", fBootable);
2164
2165 /* BIOS configuration values, first SCSI controller only. */
2166 if ( !pBusMgr->hasPCIDevice("lsilogic", 0)
2167 && !pBusMgr->hasPCIDevice("buslogic", 0)
2168 && !pBusMgr->hasPCIDevice("lsilogicsas", 1)
2169 && pBiosCfg)
2170 {
2171 InsertConfigString(pBiosCfg, "ScsiHardDiskDevice", "lsilogicsas");
2172 hrc = SetBiosDiskInfo(pMachine, pCfg, pBiosCfg, controllerName, apszBiosConfigScsi); H();
2173 }
2174
2175 ULONG cPorts = 0;
2176 hrc = ctrls[i]->COMGETTER(PortCount)(&cPorts); H();
2177 InsertConfigInteger(pCfg, "NumPorts", cPorts);
2178
2179 /* Attach the status driver */
2180 Assert(cLedSas >= 8);
2181 i_attachStatusDriver(pCtlInst, &mapStorageLeds[iLedSas], 0, 7,
2182 &mapMediumAttachments, pszCtrlDev, ulInstance);
2183 paLedDevType = &maStorageDevType[iLedSas];
2184 break;
2185 }
2186
2187 case StorageControllerType_USB:
2188 {
2189 if (pUsbDevices)
2190 {
2191 /*
2192 * USB MSDs are handled a bit different as the device instance
2193 * doesn't match the storage controller instance but the port.
2194 */
2195 InsertConfigNode(pUsbDevices, "Msd", &pDev);
2196 pCtlInst = pDev;
2197 }
2198 else
2199 return VMR3SetError(pUVM, VERR_NOT_FOUND, RT_SRC_POS,
2200 N_("There is no USB controller enabled but there\n"
2201 "is at least one USB storage device configured for this VM.\n"
2202 "To fix this problem either enable the USB controller or remove\n"
2203 "the storage device from the VM"));
2204 break;
2205 }
2206
2207 default:
2208 AssertLogRelMsgFailedReturn(("invalid storage controller type: %d\n", enmCtrlType), VERR_MAIN_CONFIG_CONSTRUCTOR_IPE);
2209 }
2210
2211 /* Attach the media to the storage controllers. */
2212 com::SafeIfaceArray<IMediumAttachment> atts;
2213 hrc = pMachine->GetMediumAttachmentsOfController(controllerName.raw(),
2214 ComSafeArrayAsOutParam(atts)); H();
2215
2216 /* Builtin I/O cache - per device setting. */
2217 BOOL fBuiltinIOCache = true;
2218 hrc = pMachine->COMGETTER(IOCacheEnabled)(&fBuiltinIOCache); H();
2219
2220
2221 for (size_t j = 0; j < atts.size(); ++j)
2222 {
2223 IMediumAttachment *pMediumAtt = atts[j];
2224 rc = i_configMediumAttachment(pszCtrlDev,
2225 ulInstance,
2226 enmBus,
2227 !!fUseHostIOCache,
2228 !!fBuiltinIOCache,
2229 false /* fSetupMerge */,
2230 0 /* uMergeSource */,
2231 0 /* uMergeTarget */,
2232 pMediumAtt,
2233 mMachineState,
2234 NULL /* phrc */,
2235 false /* fAttachDetach */,
2236 false /* fForceUnmount */,
2237 false /* fHotplug */,
2238 pUVM,
2239 paLedDevType,
2240 NULL /* ppLunL0 */);
2241 if (RT_FAILURE(rc))
2242 return rc;
2243 }
2244 H();
2245 }
2246 H();
2247
2248 /*
2249 * Network adapters
2250 */
2251#ifdef VMWARE_NET_IN_SLOT_11
2252 bool fSwapSlots3and11 = false;
2253#endif
2254 PCFGMNODE pDevPCNet = NULL; /* PCNet-type devices */
2255 InsertConfigNode(pDevices, "pcnet", &pDevPCNet);
2256#ifdef VBOX_WITH_E1000
2257 PCFGMNODE pDevE1000 = NULL; /* E1000-type devices */
2258 InsertConfigNode(pDevices, "e1000", &pDevE1000);
2259#endif
2260#ifdef VBOX_WITH_VIRTIO
2261 PCFGMNODE pDevVirtioNet = NULL; /* Virtio network devices */
2262 InsertConfigNode(pDevices, "virtio-net", &pDevVirtioNet);
2263#endif /* VBOX_WITH_VIRTIO */
2264 std::list<BootNic> llBootNics;
2265 for (ULONG ulInstance = 0; ulInstance < maxNetworkAdapters; ++ulInstance)
2266 {
2267 ComPtr<INetworkAdapter> networkAdapter;
2268 hrc = pMachine->GetNetworkAdapter(ulInstance, networkAdapter.asOutParam()); H();
2269 BOOL fEnabledNetAdapter = FALSE;
2270 hrc = networkAdapter->COMGETTER(Enabled)(&fEnabledNetAdapter); H();
2271 if (!fEnabledNetAdapter)
2272 continue;
2273
2274 /*
2275 * The virtual hardware type. Create appropriate device first.
2276 */
2277 const char *pszAdapterName = "pcnet";
2278 NetworkAdapterType_T adapterType;
2279 hrc = networkAdapter->COMGETTER(AdapterType)(&adapterType); H();
2280 switch (adapterType)
2281 {
2282 case NetworkAdapterType_Am79C970A:
2283 case NetworkAdapterType_Am79C973:
2284 pDev = pDevPCNet;
2285 break;
2286#ifdef VBOX_WITH_E1000
2287 case NetworkAdapterType_I82540EM:
2288 case NetworkAdapterType_I82543GC:
2289 case NetworkAdapterType_I82545EM:
2290 pDev = pDevE1000;
2291 pszAdapterName = "e1000";
2292 break;
2293#endif
2294#ifdef VBOX_WITH_VIRTIO
2295 case NetworkAdapterType_Virtio:
2296 pDev = pDevVirtioNet;
2297 pszAdapterName = "virtio-net";
2298 break;
2299#endif /* VBOX_WITH_VIRTIO */
2300 default:
2301 AssertMsgFailed(("Invalid network adapter type '%d' for slot '%d'",
2302 adapterType, ulInstance));
2303 return VMR3SetError(pUVM, VERR_INVALID_PARAMETER, RT_SRC_POS,
2304 N_("Invalid network adapter type '%d' for slot '%d'"),
2305 adapterType, ulInstance);
2306 }
2307
2308 InsertConfigNode(pDev, Utf8StrFmt("%u", ulInstance).c_str(), &pInst);
2309 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
2310 /* the first network card gets the PCI ID 3, the next 3 gets 8..10,
2311 * next 4 get 16..19. */
2312 int iPCIDeviceNo;
2313 switch (ulInstance)
2314 {
2315 case 0:
2316 iPCIDeviceNo = 3;
2317 break;
2318 case 1: case 2: case 3:
2319 iPCIDeviceNo = ulInstance - 1 + 8;
2320 break;
2321 case 4: case 5: case 6: case 7:
2322 iPCIDeviceNo = ulInstance - 4 + 16;
2323 break;
2324 default:
2325 /* auto assignment */
2326 iPCIDeviceNo = -1;
2327 break;
2328 }
2329#ifdef VMWARE_NET_IN_SLOT_11
2330 /*
2331 * Dirty hack for PCI slot compatibility with VMWare,
2332 * it assigns slot 0x11 to the first network controller.
2333 */
2334 if (iPCIDeviceNo == 3 && adapterType == NetworkAdapterType_I82545EM)
2335 {
2336 iPCIDeviceNo = 0x11;
2337 fSwapSlots3and11 = true;
2338 }
2339 else if (iPCIDeviceNo == 0x11 && fSwapSlots3and11)
2340 iPCIDeviceNo = 3;
2341#endif
2342 PCIBusAddress PCIAddr = PCIBusAddress(0, iPCIDeviceNo, 0);
2343 hrc = pBusMgr->assignPCIDevice(pszAdapterName, pInst, PCIAddr); H();
2344
2345 InsertConfigNode(pInst, "Config", &pCfg);
2346#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE /* not safe here yet. */ /** @todo Make PCNet ring-0 safe on 32-bit mac kernels! */
2347 if (pDev == pDevPCNet)
2348 {
2349 InsertConfigInteger(pCfg, "R0Enabled", false);
2350 }
2351#endif
2352 /*
2353 * Collect information needed for network booting and add it to the list.
2354 */
2355 BootNic nic;
2356
2357 nic.mInstance = ulInstance;
2358 /* Could be updated by reference, if auto assigned */
2359 nic.mPCIAddress = PCIAddr;
2360
2361 hrc = networkAdapter->COMGETTER(BootPriority)(&nic.mBootPrio); H();
2362
2363 llBootNics.push_back(nic);
2364
2365 /*
2366 * The virtual hardware type. PCNet supports two types.
2367 */
2368 switch (adapterType)
2369 {
2370 case NetworkAdapterType_Am79C970A:
2371 InsertConfigInteger(pCfg, "Am79C973", 0);
2372 break;
2373 case NetworkAdapterType_Am79C973:
2374 InsertConfigInteger(pCfg, "Am79C973", 1);
2375 break;
2376 case NetworkAdapterType_I82540EM:
2377 InsertConfigInteger(pCfg, "AdapterType", 0);
2378 break;
2379 case NetworkAdapterType_I82543GC:
2380 InsertConfigInteger(pCfg, "AdapterType", 1);
2381 break;
2382 case NetworkAdapterType_I82545EM:
2383 InsertConfigInteger(pCfg, "AdapterType", 2);
2384 break;
2385 }
2386
2387 /*
2388 * Get the MAC address and convert it to binary representation
2389 */
2390 Bstr macAddr;
2391 hrc = networkAdapter->COMGETTER(MACAddress)(macAddr.asOutParam()); H();
2392 Assert(!macAddr.isEmpty());
2393 Utf8Str macAddrUtf8 = macAddr;
2394 char *macStr = (char*)macAddrUtf8.c_str();
2395 Assert(strlen(macStr) == 12);
2396 RTMAC Mac;
2397 RT_ZERO(Mac);
2398 char *pMac = (char*)&Mac;
2399 for (uint32_t i = 0; i < 6; ++i)
2400 {
2401 char c1 = *macStr++ - '0';
2402 if (c1 > 9)
2403 c1 -= 7;
2404 char c2 = *macStr++ - '0';
2405 if (c2 > 9)
2406 c2 -= 7;
2407 *pMac++ = ((c1 & 0x0f) << 4) | (c2 & 0x0f);
2408 }
2409 InsertConfigBytes(pCfg, "MAC", &Mac, sizeof(Mac));
2410
2411 /*
2412 * Check if the cable is supposed to be unplugged
2413 */
2414 BOOL fCableConnected;
2415 hrc = networkAdapter->COMGETTER(CableConnected)(&fCableConnected); H();
2416 InsertConfigInteger(pCfg, "CableConnected", fCableConnected ? 1 : 0);
2417
2418 /*
2419 * Line speed to report from custom drivers
2420 */
2421 ULONG ulLineSpeed;
2422 hrc = networkAdapter->COMGETTER(LineSpeed)(&ulLineSpeed); H();
2423 InsertConfigInteger(pCfg, "LineSpeed", ulLineSpeed);
2424
2425 /*
2426 * Attach the status driver.
2427 */
2428 i_attachStatusDriver(pInst, &mapNetworkLeds[ulInstance], 0, 0, NULL, NULL, 0);
2429
2430 /*
2431 * Configure the network card now
2432 */
2433 bool fIgnoreConnectFailure = mMachineState == MachineState_Restoring;
2434 rc = i_configNetwork(pszAdapterName,
2435 ulInstance,
2436 0,
2437 networkAdapter,
2438 pCfg,
2439 pLunL0,
2440 pInst,
2441 false /*fAttachDetach*/,
2442 fIgnoreConnectFailure);
2443 if (RT_FAILURE(rc))
2444 return rc;
2445 }
2446
2447 /*
2448 * Build network boot information and transfer it to the BIOS.
2449 */
2450 if (pNetBootCfg && !llBootNics.empty()) /* NetBoot node doesn't exist for EFI! */
2451 {
2452 llBootNics.sort(); /* Sort the list by boot priority. */
2453
2454 char achBootIdx[] = "0";
2455 unsigned uBootIdx = 0;
2456
2457 for (std::list<BootNic>::iterator it = llBootNics.begin(); it != llBootNics.end(); ++it)
2458 {
2459 /* A NIC with priority 0 is only used if it's first in the list. */
2460 if (it->mBootPrio == 0 && uBootIdx != 0)
2461 break;
2462
2463 PCFGMNODE pNetBtDevCfg;
2464 achBootIdx[0] = '0' + uBootIdx++; /* Boot device order. */
2465 InsertConfigNode(pNetBootCfg, achBootIdx, &pNetBtDevCfg);
2466 InsertConfigInteger(pNetBtDevCfg, "NIC", it->mInstance);
2467 InsertConfigInteger(pNetBtDevCfg, "PCIBusNo", it->mPCIAddress.miBus);
2468 InsertConfigInteger(pNetBtDevCfg, "PCIDeviceNo", it->mPCIAddress.miDevice);
2469 InsertConfigInteger(pNetBtDevCfg, "PCIFunctionNo", it->mPCIAddress.miFn);
2470 }
2471 }
2472
2473 /*
2474 * Serial (UART) Ports
2475 */
2476 /* serial enabled mask to be passed to dev ACPI */
2477 uint16_t auSerialIoPortBase[SchemaDefs::SerialPortCount] = {0};
2478 uint8_t auSerialIrq[SchemaDefs::SerialPortCount] = {0};
2479 InsertConfigNode(pDevices, "serial", &pDev);
2480 for (ULONG ulInstance = 0; ulInstance < SchemaDefs::SerialPortCount; ++ulInstance)
2481 {
2482 ComPtr<ISerialPort> serialPort;
2483 hrc = pMachine->GetSerialPort(ulInstance, serialPort.asOutParam()); H();
2484 BOOL fEnabledSerPort = FALSE;
2485 if (serialPort)
2486 hrc = serialPort->COMGETTER(Enabled)(&fEnabledSerPort); H();
2487 if (!fEnabledSerPort)
2488 continue;
2489
2490 InsertConfigNode(pDev, Utf8StrFmt("%u", ulInstance).c_str(), &pInst);
2491 InsertConfigNode(pInst, "Config", &pCfg);
2492
2493 ULONG ulIRQ;
2494 hrc = serialPort->COMGETTER(IRQ)(&ulIRQ); H();
2495 InsertConfigInteger(pCfg, "IRQ", ulIRQ);
2496 auSerialIrq[ulInstance] = (uint8_t)ulIRQ;
2497
2498 ULONG ulIOBase;
2499 hrc = serialPort->COMGETTER(IOBase)(&ulIOBase); H();
2500 InsertConfigInteger(pCfg, "IOBase", ulIOBase);
2501 auSerialIoPortBase[ulInstance] = (uint16_t)ulIOBase;
2502
2503 BOOL fServer;
2504 hrc = serialPort->COMGETTER(Server)(&fServer); H();
2505 hrc = serialPort->COMGETTER(Path)(bstr.asOutParam()); H();
2506 PortMode_T eHostMode;
2507 hrc = serialPort->COMGETTER(HostMode)(&eHostMode); H();
2508 if (eHostMode != PortMode_Disconnected)
2509 {
2510 InsertConfigNode(pInst, "LUN#0", &pLunL0);
2511 if (eHostMode == PortMode_HostPipe)
2512 {
2513 InsertConfigString(pLunL0, "Driver", "Char");
2514 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
2515 InsertConfigString(pLunL1, "Driver", "NamedPipe");
2516 InsertConfigNode(pLunL1, "Config", &pLunL2);
2517 InsertConfigString(pLunL2, "Location", bstr);
2518 InsertConfigInteger(pLunL2, "IsServer", fServer);
2519 }
2520 else if (eHostMode == PortMode_HostDevice)
2521 {
2522 InsertConfigString(pLunL0, "Driver", "Host Serial");
2523 InsertConfigNode(pLunL0, "Config", &pLunL1);
2524 InsertConfigString(pLunL1, "DevicePath", bstr);
2525 }
2526 else if (eHostMode == PortMode_RawFile)
2527 {
2528 InsertConfigString(pLunL0, "Driver", "Char");
2529 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
2530 InsertConfigString(pLunL1, "Driver", "RawFile");
2531 InsertConfigNode(pLunL1, "Config", &pLunL2);
2532 InsertConfigString(pLunL2, "Location", bstr);
2533 }
2534 }
2535 }
2536
2537 /*
2538 * Parallel (LPT) Ports
2539 */
2540 InsertConfigNode(pDevices, "parallel", &pDev);
2541 for (ULONG ulInstance = 0; ulInstance < SchemaDefs::ParallelPortCount; ++ulInstance)
2542 {
2543 ComPtr<IParallelPort> parallelPort;
2544 hrc = pMachine->GetParallelPort(ulInstance, parallelPort.asOutParam()); H();
2545 BOOL fEnabledParPort = FALSE;
2546 if (parallelPort)
2547 {
2548 hrc = parallelPort->COMGETTER(Enabled)(&fEnabledParPort); H();
2549 }
2550 if (!fEnabledParPort)
2551 continue;
2552
2553 InsertConfigNode(pDev, Utf8StrFmt("%u", ulInstance).c_str(), &pInst);
2554 InsertConfigNode(pInst, "Config", &pCfg);
2555
2556 ULONG ulIRQ;
2557 hrc = parallelPort->COMGETTER(IRQ)(&ulIRQ); H();
2558 InsertConfigInteger(pCfg, "IRQ", ulIRQ);
2559 ULONG ulIOBase;
2560 hrc = parallelPort->COMGETTER(IOBase)(&ulIOBase); H();
2561 InsertConfigInteger(pCfg, "IOBase", ulIOBase);
2562 InsertConfigNode(pInst, "LUN#0", &pLunL0);
2563 InsertConfigString(pLunL0, "Driver", "HostParallel");
2564 InsertConfigNode(pLunL0, "Config", &pLunL1);
2565 hrc = parallelPort->COMGETTER(Path)(bstr.asOutParam()); H();
2566 InsertConfigString(pLunL1, "DevicePath", bstr);
2567 }
2568
2569 /*
2570 * VMM Device
2571 */
2572 InsertConfigNode(pDevices, "VMMDev", &pDev);
2573 InsertConfigNode(pDev, "0", &pInst);
2574 InsertConfigNode(pInst, "Config", &pCfg);
2575 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
2576 hrc = pBusMgr->assignPCIDevice("VMMDev", pInst); H();
2577
2578 Bstr hwVersion;
2579 hrc = pMachine->COMGETTER(HardwareVersion)(hwVersion.asOutParam()); H();
2580 InsertConfigInteger(pCfg, "RamSize", cbRam);
2581 if (hwVersion.compare(Bstr("1").raw()) == 0) /* <= 2.0.x */
2582 InsertConfigInteger(pCfg, "HeapEnabled", 0);
2583 Bstr snapshotFolder;
2584 hrc = pMachine->COMGETTER(SnapshotFolder)(snapshotFolder.asOutParam()); H();
2585 InsertConfigString(pCfg, "GuestCoreDumpDir", snapshotFolder);
2586
2587 /* the VMM device's Main driver */
2588 InsertConfigNode(pInst, "LUN#0", &pLunL0);
2589 InsertConfigString(pLunL0, "Driver", "HGCM");
2590 InsertConfigNode(pLunL0, "Config", &pCfg);
2591 InsertConfigInteger(pCfg, "Object", (uintptr_t)pVMMDev);
2592
2593 /*
2594 * Attach the status driver.
2595 */
2596 i_attachStatusDriver(pInst, &mapSharedFolderLed, 0, 0, NULL, NULL, 0);
2597
2598#ifndef VBOX_WITH_PDM_AUDIO_DRIVER
2599 /*
2600 * Audio Sniffer Device
2601 */
2602 InsertConfigNode(pDevices, "AudioSniffer", &pDev);
2603 InsertConfigNode(pDev, "0", &pInst);
2604 InsertConfigNode(pInst, "Config", &pCfg);
2605
2606 /* the Audio Sniffer device's Main driver */
2607 InsertConfigNode(pInst, "LUN#0", &pLunL0);
2608 InsertConfigString(pLunL0, "Driver", "MainAudioSniffer");
2609 InsertConfigNode(pLunL0, "Config", &pCfg);
2610 AudioSniffer *pAudioSniffer = mAudioSniffer;
2611 InsertConfigInteger(pCfg, "Object", (uintptr_t)pAudioSniffer);
2612#endif
2613 /*
2614 * AC'97 ICH / SoundBlaster16 audio / Intel HD Audio.
2615 */
2616 BOOL fAudioEnabled = FALSE;
2617 ComPtr<IAudioAdapter> audioAdapter;
2618 hrc = pMachine->COMGETTER(AudioAdapter)(audioAdapter.asOutParam()); H();
2619 if (audioAdapter)
2620 hrc = audioAdapter->COMGETTER(Enabled)(&fAudioEnabled); H();
2621
2622 if (fAudioEnabled)
2623 {
2624 AudioControllerType_T audioController;
2625 hrc = audioAdapter->COMGETTER(AudioController)(&audioController); H();
2626 switch (audioController)
2627 {
2628 case AudioControllerType_AC97:
2629 {
2630 /* Default: ICH AC97. */
2631 InsertConfigNode(pDevices, "ichac97", &pDev);
2632 InsertConfigNode(pDev, "0", &pInst);
2633 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
2634 hrc = pBusMgr->assignPCIDevice("ichac97", pInst); H();
2635 InsertConfigNode(pInst, "Config", &pCfg);
2636 break;
2637 }
2638 case AudioControllerType_SB16:
2639 {
2640 /* Legacy SoundBlaster16. */
2641 InsertConfigNode(pDevices, "sb16", &pDev);
2642 InsertConfigNode(pDev, "0", &pInst);
2643 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
2644 InsertConfigNode(pInst, "Config", &pCfg);
2645 InsertConfigInteger(pCfg, "IRQ", 5);
2646 InsertConfigInteger(pCfg, "DMA", 1);
2647 InsertConfigInteger(pCfg, "DMA16", 5);
2648 InsertConfigInteger(pCfg, "Port", 0x220);
2649 InsertConfigInteger(pCfg, "Version", 0x0405);
2650 break;
2651 }
2652 case AudioControllerType_HDA:
2653 {
2654 /* Intel HD Audio. */
2655 InsertConfigNode(pDevices, "hda", &pDev);
2656 InsertConfigNode(pDev, "0", &pInst);
2657 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
2658 hrc = pBusMgr->assignPCIDevice("hda", pInst); H();
2659 InsertConfigNode(pInst, "Config", &pCfg);
2660 }
2661 }
2662
2663 /* The audio driver. */
2664 InsertConfigNode(pInst, "LUN#0", &pLunL0);
2665 InsertConfigString(pLunL0, "Driver", "AUDIO");
2666 InsertConfigNode(pLunL0, "Config", &pCfg);
2667
2668#ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2669 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
2670 InsertConfigNode(pLunL1, "Config", &pCfg);
2671#endif
2672 AudioDriverType_T audioDriver;
2673 hrc = audioAdapter->COMGETTER(AudioDriver)(&audioDriver); H();
2674 switch (audioDriver)
2675 {
2676 case AudioDriverType_Null:
2677 {
2678#ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2679 InsertConfigString(pLunL1, "Driver", "NullAudio");
2680#else
2681 InsertConfigString(pCfg, "AudioDriver", "null");
2682#endif
2683 break;
2684 }
2685#ifdef RT_OS_WINDOWS
2686# ifdef VBOX_WITH_WINMM
2687 case AudioDriverType_WinMM:
2688 {
2689# ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2690 #error "Port WinMM audio backend!" /** @todo Still needed? */
2691# else
2692 InsertConfigString(pCfg, "AudioDriver", "winmm");
2693# endif
2694 break;
2695 }
2696# endif
2697 case AudioDriverType_DirectSound:
2698 {
2699#ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2700 InsertConfigString(pLunL1, "Driver", "DSoundAudio");
2701#else
2702 InsertConfigString(pCfg, "AudioDriver", "dsound");
2703#endif
2704 break;
2705 }
2706#endif /* RT_OS_WINDOWS */
2707#ifdef RT_OS_SOLARIS
2708 case AudioDriverType_SolAudio:
2709 {
2710# ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2711 /** @todo Hack alert: Find a better solution. */
2712 LogRel(("Audio: WARNING: Solaris Audio is deprecated, please switch to OSS!\n"));
2713 LogRel(("Audio: Automatically setting host audio backend to OSS\n"));
2714 /* Manually set backend to OSS for now. */
2715 InsertConfigString(pLunL1, "Driver", "OSSAudio");
2716# else
2717 InsertConfigString(pCfg, "AudioDriver", "solaudio");
2718# endif
2719 break;
2720 }
2721#endif
2722#ifdef VBOX_WITH_ALSA
2723 case AudioDriverType_ALSA:
2724 {
2725# ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2726 InsertConfigString(pLunL1, "Driver", "ALSAAudio");
2727# else
2728 InsertConfigString(pCfg, "AudioDriver", "alsa");
2729# endif
2730 break;
2731 }
2732#endif
2733#ifdef VBOX_WITH_PULSE
2734 case AudioDriverType_Pulse:
2735 {
2736# ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2737 InsertConfigString(pLunL1, "Driver", "PulseAudio");
2738# else
2739 InsertConfigString(pCfg, "AudioDriver", "pulse");
2740# endif
2741 break;
2742 }
2743#endif
2744#if defined(RT_OS_LINUX) || defined(RT_OS_FREEBSD) || defined(VBOX_WITH_SOLARIS_OSS)
2745 case AudioDriverType_OSS:
2746 {
2747# ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2748 InsertConfigString(pLunL1, "Driver", "OSSAudio");
2749# else
2750 InsertConfigString(pCfg, "AudioDriver", "ossaudio");
2751# endif
2752 break;
2753 }
2754#endif
2755#ifdef RT_OS_DARWIN
2756 case AudioDriverType_CoreAudio:
2757 {
2758# ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2759 InsertConfigString(pLunL1, "Driver", "CoreAudio");
2760# else
2761 InsertConfigString(pCfg, "AudioDriver", "coreaudio");
2762# endif
2763 break;
2764 }
2765#endif
2766 }
2767
2768 hrc = pMachine->COMGETTER(Name)(bstr.asOutParam()); H();
2769
2770#ifdef VBOX_WITH_PDM_AUDIO_DRIVER
2771 /*
2772 * The VRDE audio backend driver. This one always is there
2773 * and therefore is hardcoded here.
2774 */
2775 InsertConfigNode(pInst, "LUN#1", &pLunL1);
2776 InsertConfigString(pLunL1, "Driver", "AUDIO");
2777
2778 InsertConfigNode(pLunL1, "AttachedDriver", &pLunL1);
2779 InsertConfigString(pLunL1, "Driver", "AudioVRDE");
2780
2781 InsertConfigNode(pLunL1, "Config", &pCfg);
2782 InsertConfigString(pCfg, "AudioDriver", "AudioVRDE");
2783 InsertConfigString(pCfg, "StreamName", bstr);
2784 InsertConfigInteger(pCfg, "Object", (uintptr_t)mAudioVRDE);
2785 InsertConfigInteger(pCfg, "ObjectVRDPServer", (uintptr_t)mConsoleVRDPServer);
2786
2787 /** @todo Add audio video recording driver here. */
2788#endif
2789 }
2790
2791 /*
2792 * Shared Clipboard.
2793 */
2794 {
2795 ClipboardMode_T mode = ClipboardMode_Disabled;
2796 hrc = pMachine->COMGETTER(ClipboardMode)(&mode); H();
2797
2798 if (/* mode != ClipboardMode_Disabled */ true)
2799 {
2800 /* Load the service */
2801 rc = pVMMDev->hgcmLoadService("VBoxSharedClipboard", "VBoxSharedClipboard");
2802 if (RT_FAILURE(rc))
2803 {
2804 LogRel(("Shared clipboard is not available, rc=%Rrc\n", rc));
2805 /* That is not a fatal failure. */
2806 rc = VINF_SUCCESS;
2807 }
2808 else
2809 {
2810 LogRel(("Shared clipboard service loaded\n"));
2811
2812 i_changeClipboardMode(mode);
2813
2814 /* Setup the service. */
2815 VBOXHGCMSVCPARM parm;
2816 parm.type = VBOX_HGCM_SVC_PARM_32BIT;
2817 parm.setUInt32(!i_useHostClipboard());
2818 pVMMDev->hgcmHostCall("VBoxSharedClipboard",
2819 VBOX_SHARED_CLIPBOARD_HOST_FN_SET_HEADLESS, 1, &parm);
2820 }
2821 }
2822 }
2823
2824 /*
2825 * HGCM HostChannel.
2826 */
2827 {
2828 Bstr value;
2829 hrc = pMachine->GetExtraData(Bstr("HGCM/HostChannel").raw(),
2830 value.asOutParam());
2831
2832 if ( hrc == S_OK
2833 && value == "1")
2834 {
2835 rc = pVMMDev->hgcmLoadService("VBoxHostChannel", "VBoxHostChannel");
2836 if (RT_FAILURE(rc))
2837 {
2838 LogRel(("VBoxHostChannel is not available, rc=%Rrc\n", rc));
2839 /* That is not a fatal failure. */
2840 rc = VINF_SUCCESS;
2841 }
2842 }
2843 }
2844
2845#ifdef VBOX_WITH_DRAG_AND_DROP
2846 /*
2847 * Drag'n Drop.
2848 */
2849 {
2850 DnDMode_T enmMode = DnDMode_Disabled;
2851 hrc = pMachine->COMGETTER(DnDMode)(&enmMode); H();
2852
2853 /* Load the service */
2854 rc = pVMMDev->hgcmLoadService("VBoxDragAndDropSvc", "VBoxDragAndDropSvc");
2855 if (RT_FAILURE(rc))
2856 {
2857 LogRel(("Drag'n drop service is not available, rc=%Rrc\n", rc));
2858 /* That is not a fatal failure. */
2859 rc = VINF_SUCCESS;
2860 }
2861 else
2862 {
2863 HGCMSVCEXTHANDLE hDummy;
2864 rc = HGCMHostRegisterServiceExtension(&hDummy, "VBoxDragAndDropSvc",
2865 &GuestDnD::notifyDnDDispatcher,
2866 GuestDnDInst());
2867 if (RT_FAILURE(rc))
2868 Log(("Cannot register VBoxDragAndDropSvc extension, rc=%Rrc\n", rc));
2869 else
2870 {
2871 LogRel(("Drag'n drop service loaded\n"));
2872 rc = i_changeDnDMode(enmMode);
2873 }
2874 }
2875 }
2876#endif /* VBOX_WITH_DRAG_AND_DROP */
2877
2878#ifdef VBOX_WITH_CROGL
2879 /*
2880 * crOpenGL.
2881 */
2882 {
2883 BOOL fEnabled3D = false;
2884 hrc = pMachine->COMGETTER(Accelerate3DEnabled)(&fEnabled3D); H();
2885
2886 if ( fEnabled3D
2887# ifdef VBOX_WITH_VMSVGA3D
2888 && enmGraphicsController == GraphicsControllerType_VBoxVGA
2889# endif
2890 )
2891 {
2892 BOOL fSupports3D = VBoxOglIs3DAccelerationSupported();
2893 if (!fSupports3D)
2894 return VMR3SetError(pUVM, VERR_NOT_AVAILABLE, RT_SRC_POS,
2895 N_("This VM was configured to use 3D acceleration. However, the "
2896 "3D support of the host is not working properly and the "
2897 "VM cannot be started. To fix this problem, either "
2898 "fix the host 3D support (update the host graphics driver?) "
2899 "or disable 3D acceleration in the VM settings"));
2900
2901 /* Load the service. */
2902 rc = pVMMDev->hgcmLoadService("VBoxSharedCrOpenGL", "VBoxSharedCrOpenGL");
2903 if (RT_FAILURE(rc))
2904 {
2905 LogRel(("Failed to load Shared OpenGL service, rc=%Rrc\n", rc));
2906 /* That is not a fatal failure. */
2907 rc = VINF_SUCCESS;
2908 }
2909 else
2910 {
2911 LogRel(("Shared crOpenGL service loaded\n"));
2912
2913 /* Setup the service. */
2914 VBOXHGCMSVCPARM parm;
2915 parm.type = VBOX_HGCM_SVC_PARM_PTR;
2916
2917 parm.u.pointer.addr = (IConsole *)(Console *)this;
2918 parm.u.pointer.size = sizeof(IConsole *);
2919
2920 rc = pVMMDev->hgcmHostCall("VBoxSharedCrOpenGL", SHCRGL_HOST_FN_SET_CONSOLE,
2921 SHCRGL_CPARMS_SET_CONSOLE, &parm);
2922 if (!RT_SUCCESS(rc))
2923 AssertMsgFailed(("SHCRGL_HOST_FN_SET_CONSOLE failed with %Rrc\n", rc));
2924
2925 parm.u.pointer.addr = pVM;
2926 parm.u.pointer.size = sizeof(pVM);
2927 rc = pVMMDev->hgcmHostCall("VBoxSharedCrOpenGL",
2928 SHCRGL_HOST_FN_SET_VM, SHCRGL_CPARMS_SET_VM, &parm);
2929 if (!RT_SUCCESS(rc))
2930 AssertMsgFailed(("SHCRGL_HOST_FN_SET_VM failed with %Rrc\n", rc));
2931 }
2932 }
2933 }
2934#endif
2935
2936#ifdef VBOX_WITH_GUEST_PROPS
2937 /*
2938 * Guest property service.
2939 */
2940 rc = i_configGuestProperties(this, pUVM);
2941#endif /* VBOX_WITH_GUEST_PROPS defined */
2942
2943#ifdef VBOX_WITH_GUEST_CONTROL
2944 /*
2945 * Guest control service.
2946 */
2947 rc = i_configGuestControl(this);
2948#endif /* VBOX_WITH_GUEST_CONTROL defined */
2949
2950 /*
2951 * ACPI
2952 */
2953 BOOL fACPI;
2954 hrc = biosSettings->COMGETTER(ACPIEnabled)(&fACPI); H();
2955 if (fACPI)
2956 {
2957 /* Always show the CPU leafs when we have multiple VCPUs or when the IO-APIC is enabled.
2958 * The Windows SMP kernel needs a CPU leaf or else its idle loop will burn cpu cycles; the
2959 * intelppm driver refuses to register an idle state handler.
2960 * Always show CPU leafs for OS X guests. */
2961 BOOL fShowCpu = fOsXGuest;
2962 if (cCpus > 1 || fIOAPIC)
2963 fShowCpu = true;
2964
2965 BOOL fCpuHotPlug;
2966 hrc = pMachine->COMGETTER(CPUHotPlugEnabled)(&fCpuHotPlug); H();
2967
2968 InsertConfigNode(pDevices, "acpi", &pDev);
2969 InsertConfigNode(pDev, "0", &pInst);
2970 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
2971 InsertConfigNode(pInst, "Config", &pCfg);
2972 hrc = pBusMgr->assignPCIDevice("acpi", pInst); H();
2973
2974 InsertConfigInteger(pCfg, "RamSize", cbRam);
2975 InsertConfigInteger(pCfg, "RamHoleSize", cbRamHole);
2976 InsertConfigInteger(pCfg, "NumCPUs", cCpus);
2977
2978 InsertConfigInteger(pCfg, "IOAPIC", fIOAPIC);
2979 InsertConfigInteger(pCfg, "FdcEnabled", fFdcEnabled);
2980 InsertConfigInteger(pCfg, "HpetEnabled", fHPETEnabled);
2981 InsertConfigInteger(pCfg, "SmcEnabled", fSmcEnabled);
2982 InsertConfigInteger(pCfg, "ShowRtc", fShowRtc);
2983 if (fOsXGuest && !llBootNics.empty())
2984 {
2985 BootNic aNic = llBootNics.front();
2986 uint32_t u32NicPCIAddr = (aNic.mPCIAddress.miDevice << 16) | aNic.mPCIAddress.miFn;
2987 InsertConfigInteger(pCfg, "NicPciAddress", u32NicPCIAddr);
2988 }
2989 if (fOsXGuest && fAudioEnabled)
2990 {
2991 PCIBusAddress Address;
2992 if (pBusMgr->findPCIAddress("hda", 0, Address))
2993 {
2994 uint32_t u32AudioPCIAddr = (Address.miDevice << 16) | Address.miFn;
2995 InsertConfigInteger(pCfg, "AudioPciAddress", u32AudioPCIAddr);
2996 }
2997 }
2998 InsertConfigInteger(pCfg, "IocPciAddress", uIocPCIAddress);
2999 if (chipsetType == ChipsetType_ICH9)
3000 {
3001 InsertConfigInteger(pCfg, "McfgBase", uMcfgBase);
3002 InsertConfigInteger(pCfg, "McfgLength", cbMcfgLength);
3003 }
3004 InsertConfigInteger(pCfg, "HostBusPciAddress", uHbcPCIAddress);
3005 InsertConfigInteger(pCfg, "ShowCpu", fShowCpu);
3006 InsertConfigInteger(pCfg, "CpuHotPlug", fCpuHotPlug);
3007
3008 InsertConfigInteger(pCfg, "Serial0IoPortBase", auSerialIoPortBase[0]);
3009 InsertConfigInteger(pCfg, "Serial0Irq", auSerialIrq[0]);
3010
3011 InsertConfigInteger(pCfg, "Serial1IoPortBase", auSerialIoPortBase[1]);
3012 InsertConfigInteger(pCfg, "Serial1Irq", auSerialIrq[1]);
3013
3014 InsertConfigNode(pInst, "LUN#0", &pLunL0);
3015 InsertConfigString(pLunL0, "Driver", "ACPIHost");
3016 InsertConfigNode(pLunL0, "Config", &pCfg);
3017
3018 /* Attach the dummy CPU drivers */
3019 for (ULONG iCpuCurr = 1; iCpuCurr < cCpus; iCpuCurr++)
3020 {
3021 BOOL fCpuAttached = true;
3022
3023 if (fCpuHotPlug)
3024 {
3025 hrc = pMachine->GetCPUStatus(iCpuCurr, &fCpuAttached); H();
3026 }
3027
3028 if (fCpuAttached)
3029 {
3030 InsertConfigNode(pInst, Utf8StrFmt("LUN#%u", iCpuCurr).c_str(), &pLunL0);
3031 InsertConfigString(pLunL0, "Driver", "ACPICpu");
3032 InsertConfigNode(pLunL0, "Config", &pCfg);
3033 }
3034 }
3035 }
3036
3037 /*
3038 * Configure DBGF (Debug(ger) Facility).
3039 */
3040 {
3041 PCFGMNODE pDbgf;
3042 InsertConfigNode(pRoot, "DBGF", &pDbgf);
3043
3044 /* Paths to search for debug info and such things. */
3045 hrc = pMachine->COMGETTER(SettingsFilePath)(bstr.asOutParam()); H();
3046 Utf8Str strSettingsPath(bstr);
3047 bstr.setNull();
3048 strSettingsPath.stripFilename();
3049
3050 char szHomeDir[RTPATH_MAX];
3051 rc = RTPathUserHome(szHomeDir, sizeof(szHomeDir));
3052 if (RT_FAILURE(rc))
3053 szHomeDir[0] = '\0';
3054
3055 Utf8Str strPath;
3056 strPath.append(strSettingsPath).append("/debug/;");
3057 strPath.append(strSettingsPath).append("/;");
3058 strPath.append(szHomeDir).append("/");
3059
3060 InsertConfigString(pDbgf, "Path", strPath.c_str());
3061
3062 /* Tracing configuration. */
3063 BOOL fTracingEnabled;
3064 hrc = pMachine->COMGETTER(TracingEnabled)(&fTracingEnabled); H();
3065 if (fTracingEnabled)
3066 InsertConfigInteger(pDbgf, "TracingEnabled", 1);
3067
3068 hrc = pMachine->COMGETTER(TracingConfig)(bstr.asOutParam()); H();
3069 if (fTracingEnabled)
3070 InsertConfigString(pDbgf, "TracingConfig", bstr);
3071
3072 BOOL fAllowTracingToAccessVM;
3073 hrc = pMachine->COMGETTER(AllowTracingToAccessVM)(&fAllowTracingToAccessVM); H();
3074 if (fAllowTracingToAccessVM)
3075 InsertConfigInteger(pPDM, "AllowTracingToAccessVM", 1);
3076 }
3077 }
3078 catch (ConfigError &x)
3079 {
3080 // InsertConfig threw something:
3081 return x.m_vrc;
3082 }
3083 catch (HRESULT hrcXcpt)
3084 {
3085 AssertLogRelMsgFailedReturn(("hrc=%Rhrc\n", hrcXcpt), VERR_MAIN_CONFIG_CONSTRUCTOR_COM_ERROR);
3086 }
3087
3088#ifdef VBOX_WITH_EXTPACK
3089 /*
3090 * Call the extension pack hooks if everything went well thus far.
3091 */
3092 if (RT_SUCCESS(rc))
3093 {
3094 pAlock->release();
3095 rc = mptrExtPackManager->i_callAllVmConfigureVmmHooks(this, pVM);
3096 pAlock->acquire();
3097 }
3098#endif
3099
3100 /*
3101 * Apply the CFGM overlay.
3102 */
3103 if (RT_SUCCESS(rc))
3104 rc = i_configCfgmOverlay(pRoot, virtualBox, pMachine);
3105
3106 /*
3107 * Dump all extradata API settings tweaks, both global and per VM.
3108 */
3109 if (RT_SUCCESS(rc))
3110 rc = i_configDumpAPISettingsTweaks(virtualBox, pMachine);
3111
3112#undef H
3113
3114 pAlock->release(); /* Avoid triggering the lock order inversion check. */
3115
3116 /*
3117 * Register VM state change handler.
3118 */
3119 int rc2 = VMR3AtStateRegister(pUVM, Console::i_vmstateChangeCallback, this);
3120 AssertRC(rc2);
3121 if (RT_SUCCESS(rc))
3122 rc = rc2;
3123
3124 /*
3125 * Register VM runtime error handler.
3126 */
3127 rc2 = VMR3AtRuntimeErrorRegister(pUVM, Console::i_setVMRuntimeErrorCallback, this);
3128 AssertRC(rc2);
3129 if (RT_SUCCESS(rc))
3130 rc = rc2;
3131
3132 pAlock->acquire();
3133
3134 LogFlowFunc(("vrc = %Rrc\n", rc));
3135 LogFlowFuncLeave();
3136
3137 return rc;
3138}
3139
3140/**
3141 * Applies the CFGM overlay as specified by VBoxInternal/XXX extra data
3142 * values.
3143 *
3144 * @returns VBox status code.
3145 * @param pRoot The root of the configuration tree.
3146 * @param pVirtualBox Pointer to the IVirtualBox interface.
3147 * @param pMachine Pointer to the IMachine interface.
3148 */
3149/* static */
3150int Console::i_configCfgmOverlay(PCFGMNODE pRoot, IVirtualBox *pVirtualBox, IMachine *pMachine)
3151{
3152 /*
3153 * CFGM overlay handling.
3154 *
3155 * Here we check the extra data entries for CFGM values
3156 * and create the nodes and insert the values on the fly. Existing
3157 * values will be removed and reinserted. CFGM is typed, so by default
3158 * we will guess whether it's a string or an integer (byte arrays are
3159 * not currently supported). It's possible to override this autodetection
3160 * by adding "string:", "integer:" or "bytes:" (future).
3161 *
3162 * We first perform a run on global extra data, then on the machine
3163 * extra data to support global settings with local overrides.
3164 */
3165 int rc = VINF_SUCCESS;
3166 try
3167 {
3168 /** @todo add support for removing nodes and byte blobs. */
3169 /*
3170 * Get the next key
3171 */
3172 SafeArray<BSTR> aGlobalExtraDataKeys;
3173 SafeArray<BSTR> aMachineExtraDataKeys;
3174 HRESULT hrc = pVirtualBox->GetExtraDataKeys(ComSafeArrayAsOutParam(aGlobalExtraDataKeys));
3175 AssertMsg(SUCCEEDED(hrc), ("VirtualBox::GetExtraDataKeys failed with %Rhrc\n", hrc));
3176
3177 // remember the no. of global values so we can call the correct method below
3178 size_t cGlobalValues = aGlobalExtraDataKeys.size();
3179
3180 hrc = pMachine->GetExtraDataKeys(ComSafeArrayAsOutParam(aMachineExtraDataKeys));
3181 AssertMsg(SUCCEEDED(hrc), ("Machine::GetExtraDataKeys failed with %Rhrc\n", hrc));
3182
3183 // build a combined list from global keys...
3184 std::list<Utf8Str> llExtraDataKeys;
3185
3186 for (size_t i = 0; i < aGlobalExtraDataKeys.size(); ++i)
3187 llExtraDataKeys.push_back(Utf8Str(aGlobalExtraDataKeys[i]));
3188 // ... and machine keys
3189 for (size_t i = 0; i < aMachineExtraDataKeys.size(); ++i)
3190 llExtraDataKeys.push_back(Utf8Str(aMachineExtraDataKeys[i]));
3191
3192 size_t i2 = 0;
3193 for (std::list<Utf8Str>::const_iterator it = llExtraDataKeys.begin();
3194 it != llExtraDataKeys.end();
3195 ++it, ++i2)
3196 {
3197 const Utf8Str &strKey = *it;
3198
3199 /*
3200 * We only care about keys starting with "VBoxInternal/" (skip "G:" or "M:")
3201 */
3202 if (!strKey.startsWith("VBoxInternal/"))
3203 continue;
3204
3205 const char *pszExtraDataKey = strKey.c_str() + sizeof("VBoxInternal/") - 1;
3206
3207 // get the value
3208 Bstr bstrExtraDataValue;
3209 if (i2 < cGlobalValues)
3210 // this is still one of the global values:
3211 hrc = pVirtualBox->GetExtraData(Bstr(strKey).raw(),
3212 bstrExtraDataValue.asOutParam());
3213 else
3214 hrc = pMachine->GetExtraData(Bstr(strKey).raw(),
3215 bstrExtraDataValue.asOutParam());
3216 if (FAILED(hrc))
3217 LogRel(("Warning: Cannot get extra data key %s, rc = %Rhrc\n", strKey.c_str(), hrc));
3218
3219 /*
3220 * The key will be in the format "Node1/Node2/Value" or simply "Value".
3221 * Split the two and get the node, delete the value and create the node
3222 * if necessary.
3223 */
3224 PCFGMNODE pNode;
3225 const char *pszCFGMValueName = strrchr(pszExtraDataKey, '/');
3226 if (pszCFGMValueName)
3227 {
3228 /* terminate the node and advance to the value (Utf8Str might not
3229 offically like this but wtf) */
3230 *(char*)pszCFGMValueName = '\0';
3231 ++pszCFGMValueName;
3232
3233 /* does the node already exist? */
3234 pNode = CFGMR3GetChild(pRoot, pszExtraDataKey);
3235 if (pNode)
3236 CFGMR3RemoveValue(pNode, pszCFGMValueName);
3237 else
3238 {
3239 /* create the node */
3240 rc = CFGMR3InsertNode(pRoot, pszExtraDataKey, &pNode);
3241 if (RT_FAILURE(rc))
3242 {
3243 AssertLogRelMsgRC(rc, ("failed to insert node '%s'\n", pszExtraDataKey));
3244 continue;
3245 }
3246 Assert(pNode);
3247 }
3248 }
3249 else
3250 {
3251 /* root value (no node path). */
3252 pNode = pRoot;
3253 pszCFGMValueName = pszExtraDataKey;
3254 pszExtraDataKey--;
3255 CFGMR3RemoveValue(pNode, pszCFGMValueName);
3256 }
3257
3258 /*
3259 * Now let's have a look at the value.
3260 * Empty strings means that we should remove the value, which we've
3261 * already done above.
3262 */
3263 Utf8Str strCFGMValueUtf8(bstrExtraDataValue);
3264 if (!strCFGMValueUtf8.isEmpty())
3265 {
3266 uint64_t u64Value;
3267
3268 /* check for type prefix first. */
3269 if (!strncmp(strCFGMValueUtf8.c_str(), RT_STR_TUPLE("string:")))
3270 InsertConfigString(pNode, pszCFGMValueName, strCFGMValueUtf8.c_str() + sizeof("string:") - 1);
3271 else if (!strncmp(strCFGMValueUtf8.c_str(), RT_STR_TUPLE("integer:")))
3272 {
3273 rc = RTStrToUInt64Full(strCFGMValueUtf8.c_str() + sizeof("integer:") - 1, 0, &u64Value);
3274 if (RT_SUCCESS(rc))
3275 rc = CFGMR3InsertInteger(pNode, pszCFGMValueName, u64Value);
3276 }
3277 else if (!strncmp(strCFGMValueUtf8.c_str(), RT_STR_TUPLE("bytes:")))
3278 {
3279 char const *pszBase64 = strCFGMValueUtf8.c_str() + sizeof("bytes:") - 1;
3280 ssize_t cbValue = RTBase64DecodedSize(pszBase64, NULL);
3281 if (cbValue > 0)
3282 {
3283 void *pvBytes = RTMemTmpAlloc(cbValue);
3284 if (pvBytes)
3285 {
3286 rc = RTBase64Decode(pszBase64, pvBytes, cbValue, NULL, NULL);
3287 if (RT_SUCCESS(rc))
3288 rc = CFGMR3InsertBytes(pNode, pszCFGMValueName, pvBytes, cbValue);
3289 RTMemTmpFree(pvBytes);
3290 }
3291 else
3292 rc = VERR_NO_TMP_MEMORY;
3293 }
3294 else if (cbValue == 0)
3295 rc = CFGMR3InsertBytes(pNode, pszCFGMValueName, NULL, 0);
3296 else
3297 rc = VERR_INVALID_BASE64_ENCODING;
3298 }
3299 /* auto detect type. */
3300 else if (RT_SUCCESS(RTStrToUInt64Full(strCFGMValueUtf8.c_str(), 0, &u64Value)))
3301 rc = CFGMR3InsertInteger(pNode, pszCFGMValueName, u64Value);
3302 else
3303 InsertConfigString(pNode, pszCFGMValueName, strCFGMValueUtf8);
3304 AssertLogRelMsgRCBreak(rc, ("failed to insert CFGM value '%s' to key '%s'\n",
3305 strCFGMValueUtf8.c_str(), pszExtraDataKey));
3306 }
3307 }
3308 }
3309 catch (ConfigError &x)
3310 {
3311 // InsertConfig threw something:
3312 return x.m_vrc;
3313 }
3314 return rc;
3315}
3316
3317/**
3318 * Dumps the API settings tweaks as specified by VBoxInternal2/XXX extra data
3319 * values.
3320 *
3321 * @returns VBox status code.
3322 * @param pVirtualBox Pointer to the IVirtualBox interface.
3323 * @param pMachine Pointer to the IMachine interface.
3324 */
3325/* static */
3326int Console::i_configDumpAPISettingsTweaks(IVirtualBox *pVirtualBox, IMachine *pMachine)
3327{
3328 {
3329 SafeArray<BSTR> aGlobalExtraDataKeys;
3330 HRESULT hrc = pVirtualBox->GetExtraDataKeys(ComSafeArrayAsOutParam(aGlobalExtraDataKeys));
3331 AssertMsg(SUCCEEDED(hrc), ("VirtualBox::GetExtraDataKeys failed with %Rhrc\n", hrc));
3332 bool hasKey = false;
3333 for (size_t i = 0; i < aGlobalExtraDataKeys.size(); i++)
3334 {
3335 Utf8Str strKey(aGlobalExtraDataKeys[i]);
3336 if (!strKey.startsWith("VBoxInternal2/"))
3337 continue;
3338
3339 Bstr bstrValue;
3340 hrc = pVirtualBox->GetExtraData(Bstr(strKey).raw(),
3341 bstrValue.asOutParam());
3342 if (FAILED(hrc))
3343 continue;
3344 if (!hasKey)
3345 LogRel(("Global extradata API settings:\n"));
3346 LogRel((" %s=\"%ls\"\n", strKey.c_str(), bstrValue.raw()));
3347 hasKey = true;
3348 }
3349 }
3350
3351 {
3352 SafeArray<BSTR> aMachineExtraDataKeys;
3353 HRESULT hrc = pMachine->GetExtraDataKeys(ComSafeArrayAsOutParam(aMachineExtraDataKeys));
3354 AssertMsg(SUCCEEDED(hrc), ("Machine::GetExtraDataKeys failed with %Rhrc\n", hrc));
3355 bool hasKey = false;
3356 for (size_t i = 0; i < aMachineExtraDataKeys.size(); i++)
3357 {
3358 Utf8Str strKey(aMachineExtraDataKeys[i]);
3359 if (!strKey.startsWith("VBoxInternal2/"))
3360 continue;
3361
3362 Bstr bstrValue;
3363 hrc = pMachine->GetExtraData(Bstr(strKey).raw(),
3364 bstrValue.asOutParam());
3365 if (FAILED(hrc))
3366 continue;
3367 if (!hasKey)
3368 LogRel(("Per-VM extradata API settings:\n"));
3369 LogRel((" %s=\"%ls\"\n", strKey.c_str(), bstrValue.raw()));
3370 hasKey = true;
3371 }
3372 }
3373
3374 return VINF_SUCCESS;
3375}
3376
3377int Console::i_configGraphicsController(PCFGMNODE pDevices,
3378 const GraphicsControllerType_T enmGraphicsController,
3379 BusAssignmentManager *pBusMgr,
3380 const ComPtr<IMachine> &ptrMachine,
3381 const ComPtr<IBIOSSettings> &ptrBiosSettings,
3382 bool fHMEnabled)
3383{
3384 // InsertConfig* throws
3385 try
3386 {
3387 PCFGMNODE pDev, pInst, pCfg, pLunL0;
3388 HRESULT hrc;
3389 Bstr bstr;
3390 const char *pcszDevice = "vga";
3391
3392#define H() AssertLogRelMsgReturn(!FAILED(hrc), ("hrc=%Rhrc\n", hrc), VERR_MAIN_CONFIG_CONSTRUCTOR_COM_ERROR)
3393 InsertConfigNode(pDevices, pcszDevice, &pDev);
3394 InsertConfigNode(pDev, "0", &pInst);
3395 InsertConfigInteger(pInst, "Trusted", 1); /* boolean */
3396
3397 hrc = pBusMgr->assignPCIDevice(pcszDevice, pInst); H();
3398 InsertConfigNode(pInst, "Config", &pCfg);
3399 ULONG cVRamMBs;
3400 hrc = ptrMachine->COMGETTER(VRAMSize)(&cVRamMBs); H();
3401 InsertConfigInteger(pCfg, "VRamSize", cVRamMBs * _1M);
3402 ULONG cMonitorCount;
3403 hrc = ptrMachine->COMGETTER(MonitorCount)(&cMonitorCount); H();
3404 InsertConfigInteger(pCfg, "MonitorCount", cMonitorCount);
3405#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE
3406 InsertConfigInteger(pCfg, "R0Enabled", fHMEnabled);
3407#else
3408 NOREF(fHMEnabled);
3409#endif
3410
3411 i_attachStatusDriver(pInst, &mapCrOglLed, 0, 0, NULL, NULL, 0);
3412
3413#ifdef VBOX_WITH_VMSVGA
3414 if (enmGraphicsController == GraphicsControllerType_VMSVGA)
3415 {
3416 InsertConfigInteger(pCfg, "VMSVGAEnabled", true);
3417#ifdef VBOX_WITH_VMSVGA3D
3418 IFramebuffer *pFramebuffer = NULL;
3419 hrc = i_getDisplay()->QueryFramebuffer(0, &pFramebuffer);
3420 if (SUCCEEDED(hrc) && pFramebuffer)
3421 {
3422 LONG64 winId = 0;
3423 /* @todo deal with multimonitor setup */
3424 Assert(cMonitorCount == 1);
3425 hrc = pFramebuffer->COMGETTER(WinId)(&winId);
3426 InsertConfigInteger(pCfg, "HostWindowId", winId);
3427 pFramebuffer->Release();
3428 }
3429 BOOL f3DEnabled;
3430 hrc = ptrMachine->COMGETTER(Accelerate3DEnabled)(&f3DEnabled); H();
3431 InsertConfigInteger(pCfg, "VMSVGA3dEnabled", f3DEnabled);
3432#endif
3433 }
3434#endif
3435
3436 /* Custom VESA mode list */
3437 unsigned cModes = 0;
3438 for (unsigned iMode = 1; iMode <= 16; ++iMode)
3439 {
3440 char szExtraDataKey[sizeof("CustomVideoModeXX")];
3441 RTStrPrintf(szExtraDataKey, sizeof(szExtraDataKey), "CustomVideoMode%u", iMode);
3442 hrc = ptrMachine->GetExtraData(Bstr(szExtraDataKey).raw(), bstr.asOutParam()); H();
3443 if (bstr.isEmpty())
3444 break;
3445 InsertConfigString(pCfg, szExtraDataKey, bstr);
3446 ++cModes;
3447 }
3448 InsertConfigInteger(pCfg, "CustomVideoModes", cModes);
3449
3450 /* VESA height reduction */
3451 ULONG ulHeightReduction;
3452 IFramebuffer *pFramebuffer = NULL;
3453 hrc = i_getDisplay()->QueryFramebuffer(0, &pFramebuffer);
3454 if (SUCCEEDED(hrc) && pFramebuffer)
3455 {
3456 hrc = pFramebuffer->COMGETTER(HeightReduction)(&ulHeightReduction); H();
3457 pFramebuffer->Release();
3458 pFramebuffer = NULL;
3459 }
3460 else
3461 {
3462 /* If framebuffer is not available, there is no height reduction. */
3463 ulHeightReduction = 0;
3464 }
3465 InsertConfigInteger(pCfg, "HeightReduction", ulHeightReduction);
3466
3467 /*
3468 * BIOS logo
3469 */
3470 BOOL fFadeIn;
3471 hrc = ptrBiosSettings->COMGETTER(LogoFadeIn)(&fFadeIn); H();
3472 InsertConfigInteger(pCfg, "FadeIn", fFadeIn ? 1 : 0);
3473 BOOL fFadeOut;
3474 hrc = ptrBiosSettings->COMGETTER(LogoFadeOut)(&fFadeOut); H();
3475 InsertConfigInteger(pCfg, "FadeOut", fFadeOut ? 1: 0);
3476 ULONG logoDisplayTime;
3477 hrc = ptrBiosSettings->COMGETTER(LogoDisplayTime)(&logoDisplayTime); H();
3478 InsertConfigInteger(pCfg, "LogoTime", logoDisplayTime);
3479 Bstr logoImagePath;
3480 hrc = ptrBiosSettings->COMGETTER(LogoImagePath)(logoImagePath.asOutParam()); H();
3481 InsertConfigString(pCfg, "LogoFile", Utf8Str(!logoImagePath.isEmpty() ? logoImagePath : "") );
3482
3483 /*
3484 * Boot menu
3485 */
3486 BIOSBootMenuMode_T eBootMenuMode;
3487 int iShowBootMenu;
3488 hrc = ptrBiosSettings->COMGETTER(BootMenuMode)(&eBootMenuMode); H();
3489 switch (eBootMenuMode)
3490 {
3491 case BIOSBootMenuMode_Disabled: iShowBootMenu = 0; break;
3492 case BIOSBootMenuMode_MenuOnly: iShowBootMenu = 1; break;
3493 default: iShowBootMenu = 2; break;
3494 }
3495 InsertConfigInteger(pCfg, "ShowBootMenu", iShowBootMenu);
3496
3497 /* Attach the display. */
3498 InsertConfigNode(pInst, "LUN#0", &pLunL0);
3499 InsertConfigString(pLunL0, "Driver", "MainDisplay");
3500 InsertConfigNode(pLunL0, "Config", &pCfg);
3501 Display *pDisplay = mDisplay;
3502 InsertConfigInteger(pCfg, "Object", (uintptr_t)pDisplay);
3503 }
3504 catch (ConfigError &x)
3505 {
3506 // InsertConfig threw something:
3507 return x.m_vrc;
3508 }
3509
3510#undef H
3511
3512 return VINF_SUCCESS;
3513}
3514
3515
3516/**
3517 * Ellipsis to va_list wrapper for calling setVMRuntimeErrorCallback.
3518 */
3519void Console::i_setVMRuntimeErrorCallbackF(uint32_t fFlags, const char *pszErrorId, const char *pszFormat, ...)
3520{
3521 va_list va;
3522 va_start(va, pszFormat);
3523 i_setVMRuntimeErrorCallback(NULL, this, fFlags, pszErrorId, pszFormat, va);
3524 va_end(va);
3525}
3526
3527/* XXX introduce RT format specifier */
3528static uint64_t formatDiskSize(uint64_t u64Size, const char **pszUnit)
3529{
3530 if (u64Size > INT64_C(5000)*_1G)
3531 {
3532 *pszUnit = "TB";
3533 return u64Size / _1T;
3534 }
3535 else if (u64Size > INT64_C(5000)*_1M)
3536 {
3537 *pszUnit = "GB";
3538 return u64Size / _1G;
3539 }
3540 else
3541 {
3542 *pszUnit = "MB";
3543 return u64Size / _1M;
3544 }
3545}
3546
3547int Console::i_configMediumAttachment(const char *pcszDevice,
3548 unsigned uInstance,
3549 StorageBus_T enmBus,
3550 bool fUseHostIOCache,
3551 bool fBuiltinIOCache,
3552 bool fSetupMerge,
3553 unsigned uMergeSource,
3554 unsigned uMergeTarget,
3555 IMediumAttachment *pMediumAtt,
3556 MachineState_T aMachineState,
3557 HRESULT *phrc,
3558 bool fAttachDetach,
3559 bool fForceUnmount,
3560 bool fHotplug,
3561 PUVM pUVM,
3562 DeviceType_T *paLedDevType,
3563 PCFGMNODE *ppLunL0)
3564{
3565 // InsertConfig* throws
3566 try
3567 {
3568 int rc = VINF_SUCCESS;
3569 HRESULT hrc;
3570 Bstr bstr;
3571 PCFGMNODE pCtlInst = NULL;
3572
3573// #define RC_CHECK() AssertMsgReturn(RT_SUCCESS(rc), ("rc=%Rrc\n", rc), rc)
3574#define H() AssertLogRelMsgReturn(!FAILED(hrc), ("hrc=%Rhrc\n", hrc), VERR_MAIN_CONFIG_CONSTRUCTOR_COM_ERROR)
3575
3576 LONG lDev;
3577 hrc = pMediumAtt->COMGETTER(Device)(&lDev); H();
3578 LONG lPort;
3579 hrc = pMediumAtt->COMGETTER(Port)(&lPort); H();
3580 DeviceType_T lType;
3581 hrc = pMediumAtt->COMGETTER(Type)(&lType); H();
3582 BOOL fNonRotational;
3583 hrc = pMediumAtt->COMGETTER(NonRotational)(&fNonRotational); H();
3584 BOOL fDiscard;
3585 hrc = pMediumAtt->COMGETTER(Discard)(&fDiscard); H();
3586
3587 unsigned uLUN;
3588 PCFGMNODE pLunL0 = NULL;
3589 hrc = Console::i_convertBusPortDeviceToLun(enmBus, lPort, lDev, uLUN); H();
3590
3591 /* Determine the base path for the device instance. */
3592 if (enmBus != StorageBus_USB)
3593 pCtlInst = CFGMR3GetChildF(CFGMR3GetRootU(pUVM), "Devices/%s/%u/", pcszDevice, uInstance);
3594 else
3595 {
3596 /* If we hotplug a USB device create a new CFGM tree. */
3597 if (!fHotplug)
3598 pCtlInst = CFGMR3GetChildF(CFGMR3GetRootU(pUVM), "USB/%s/", pcszDevice, uInstance);
3599 else
3600 pCtlInst = CFGMR3CreateTree(pUVM);
3601 }
3602 AssertReturn(pCtlInst, VERR_INTERNAL_ERROR);
3603
3604 if (enmBus == StorageBus_USB)
3605 {
3606 PCFGMNODE pCfg = NULL;
3607
3608 /* Create correct instance. */
3609 if (!fHotplug && !fAttachDetach)
3610 InsertConfigNode(pCtlInst, Utf8StrFmt("%d", lPort).c_str(), &pCtlInst);
3611 else if (fAttachDetach)
3612 pCtlInst = CFGMR3GetChildF(pCtlInst, "%d/", lPort);
3613
3614 if (!fAttachDetach)
3615 InsertConfigNode(pCtlInst, "Config", &pCfg);
3616
3617 uInstance = lPort; /* Overwrite uInstance with the correct one. */
3618
3619 if (!fHotplug && !fAttachDetach)
3620 {
3621 char aszUuid[RTUUID_STR_LENGTH + 1];
3622 USBStorageDevice UsbMsd = USBStorageDevice();
3623
3624 memset(aszUuid, 0, sizeof(aszUuid));
3625 rc = RTUuidCreate(&UsbMsd.mUuid);
3626 AssertRCReturn(rc, rc);
3627 rc = RTUuidToStr(&UsbMsd.mUuid, aszUuid, sizeof(aszUuid));
3628 AssertRCReturn(rc, rc);
3629
3630 UsbMsd.iPort = uInstance;
3631
3632 InsertConfigString(pCtlInst, "UUID", aszUuid);
3633 mUSBStorageDevices.push_back(UsbMsd);
3634
3635 /** @todo: No LED after hotplugging. */
3636 /* Attach the status driver */
3637 Assert(cLedUsb >= 8);
3638 i_attachStatusDriver(pCtlInst, &mapStorageLeds[iLedUsb], 0, 7,
3639 &mapMediumAttachments, pcszDevice, 0);
3640 paLedDevType = &maStorageDevType[iLedUsb];
3641 }
3642 }
3643
3644 /* First check if the LUN already exists. */
3645 pLunL0 = CFGMR3GetChildF(pCtlInst, "LUN#%u", uLUN);
3646 if (pLunL0)
3647 {
3648 if (fAttachDetach)
3649 {
3650 if (lType != DeviceType_HardDisk)
3651 {
3652 /* Unmount existing media only for floppy and DVD drives. */
3653 PPDMIBASE pBase;
3654 if (enmBus == StorageBus_USB)
3655 rc = PDMR3UsbQueryLun(pUVM, pcszDevice, uInstance, uLUN, &pBase);
3656 else
3657 rc = PDMR3QueryLun(pUVM, pcszDevice, uInstance, uLUN, &pBase);
3658 if (RT_FAILURE(rc))
3659 {
3660 if (rc == VERR_PDM_LUN_NOT_FOUND || rc == VERR_PDM_NO_DRIVER_ATTACHED_TO_LUN)
3661 rc = VINF_SUCCESS;
3662 AssertRC(rc);
3663 }
3664 else
3665 {
3666 PPDMIMOUNT pIMount = PDMIBASE_QUERY_INTERFACE(pBase, PDMIMOUNT);
3667 AssertReturn(pIMount, VERR_INVALID_POINTER);
3668
3669 /* Unmount the media (but do not eject the medium!) */
3670 rc = pIMount->pfnUnmount(pIMount, fForceUnmount, false /*=fEject*/);
3671 if (rc == VERR_PDM_MEDIA_NOT_MOUNTED)
3672 rc = VINF_SUCCESS;
3673 /* for example if the medium is locked */
3674 else if (RT_FAILURE(rc))
3675 return rc;
3676 }
3677 }
3678
3679 if (enmBus == StorageBus_USB)
3680 rc = PDMR3UsbDriverDetach(pUVM, pcszDevice, uInstance, uLUN, NULL, 0,
3681 fHotplug ? 0 : PDM_TACH_FLAGS_NOT_HOT_PLUG);
3682 else
3683 rc = PDMR3DeviceDetach(pUVM, pcszDevice, uInstance, uLUN, fHotplug ? 0 : PDM_TACH_FLAGS_NOT_HOT_PLUG);
3684 if (rc == VERR_PDM_NO_DRIVER_ATTACHED_TO_LUN)
3685 rc = VINF_SUCCESS;
3686 AssertRCReturn(rc, rc);
3687
3688 CFGMR3RemoveNode(pLunL0);
3689 }
3690 else
3691 AssertFailedReturn(VERR_INTERNAL_ERROR);
3692 }
3693
3694 InsertConfigNode(pCtlInst, Utf8StrFmt("LUN#%u", uLUN).c_str(), &pLunL0);
3695 if (ppLunL0)
3696 *ppLunL0 = pLunL0;
3697
3698 PCFGMNODE pCfg = CFGMR3GetChild(pCtlInst, "Config");
3699 if (pCfg)
3700 {
3701 if (!strcmp(pcszDevice, "piix3ide"))
3702 {
3703 PCFGMNODE pDrive = CFGMR3GetChild(pCfg, g_apszIDEDrives[uLUN]);
3704 if (!pDrive)
3705 InsertConfigNode(pCfg, g_apszIDEDrives[uLUN], &pDrive);
3706 /* Don't use the RemoveConfigValue wrapper above, as we don't
3707 * know if the leaf is present or not. */
3708 CFGMR3RemoveValue(pDrive, "NonRotationalMedium");
3709 InsertConfigInteger(pDrive, "NonRotationalMedium", !!fNonRotational);
3710 }
3711 else if (!strcmp(pcszDevice, "ahci"))
3712 {
3713 Utf8Str strPort = Utf8StrFmt("Port%u", uLUN);
3714 PCFGMNODE pDrive = CFGMR3GetChild(pCfg, strPort.c_str());
3715 if (!pDrive)
3716 InsertConfigNode(pCfg, strPort.c_str(), &pDrive);
3717 /* Don't use the RemoveConfigValue wrapper above, as we don't
3718 * know if the leaf is present or not. */
3719 CFGMR3RemoveValue(pDrive, "NonRotationalMedium");
3720 InsertConfigInteger(pDrive, "NonRotationalMedium", !!fNonRotational);
3721 }
3722 }
3723
3724 Utf8Str devicePath = Utf8StrFmt("%s/%u/LUN#%u", pcszDevice, uInstance, uLUN);
3725 mapMediumAttachments[devicePath] = pMediumAtt;
3726
3727 /* SCSI has a another driver between device and block. */
3728 if (enmBus == StorageBus_SCSI || enmBus == StorageBus_SAS || enmBus == StorageBus_USB)
3729 {
3730 InsertConfigString(pLunL0, "Driver", "SCSI");
3731 PCFGMNODE pL1Cfg = NULL;
3732 InsertConfigNode(pLunL0, "Config", &pL1Cfg);
3733 InsertConfigInteger(pL1Cfg, "NonRotationalMedium", !!fNonRotational);
3734
3735 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL0);
3736 }
3737
3738 ComPtr<IMedium> pMedium;
3739 hrc = pMediumAtt->COMGETTER(Medium)(pMedium.asOutParam()); H();
3740
3741 /*
3742 * 1. Only check this for hard disk images.
3743 * 2. Only check during VM creation and not later, especially not during
3744 * taking an online snapshot!
3745 */
3746 if ( lType == DeviceType_HardDisk
3747 && ( aMachineState == MachineState_Starting
3748 || aMachineState == MachineState_Restoring))
3749 {
3750 /*
3751 * Some sanity checks.
3752 */
3753 ComPtr<IMediumFormat> pMediumFormat;
3754 hrc = pMedium->COMGETTER(MediumFormat)(pMediumFormat.asOutParam()); H();
3755 ULONG uCaps = 0;
3756 com::SafeArray <MediumFormatCapabilities_T> mediumFormatCap;
3757 hrc = pMediumFormat->COMGETTER(Capabilities)(ComSafeArrayAsOutParam(mediumFormatCap)); H();
3758
3759 for (ULONG j = 0; j < mediumFormatCap.size(); j++)
3760 uCaps |= mediumFormatCap[j];
3761
3762 if (uCaps & MediumFormatCapabilities_File)
3763 {
3764 Bstr strFile;
3765 hrc = pMedium->COMGETTER(Location)(strFile.asOutParam()); H();
3766 Utf8Str utfFile = Utf8Str(strFile);
3767 Bstr strSnap;
3768 ComPtr<IMachine> pMachine = i_machine();
3769 hrc = pMachine->COMGETTER(SnapshotFolder)(strSnap.asOutParam()); H();
3770 Utf8Str utfSnap = Utf8Str(strSnap);
3771 RTFSTYPE enmFsTypeFile = RTFSTYPE_UNKNOWN;
3772 RTFSTYPE enmFsTypeSnap = RTFSTYPE_UNKNOWN;
3773 int rc2 = RTFsQueryType(utfFile.c_str(), &enmFsTypeFile);
3774 AssertMsgRCReturn(rc2, ("Querying the file type of '%s' failed!\n", utfFile.c_str()), rc2);
3775 /* Ignore the error code. On error, the file system type is still 'unknown' so
3776 * none of the following paths are taken. This can happen for new VMs which
3777 * still don't have a snapshot folder. */
3778 (void)RTFsQueryType(utfSnap.c_str(), &enmFsTypeSnap);
3779 if (!mfSnapshotFolderDiskTypeShown)
3780 {
3781 LogRel(("File system of '%s' (snapshots) is %s\n",
3782 utfSnap.c_str(), RTFsTypeName(enmFsTypeSnap)));
3783 mfSnapshotFolderDiskTypeShown = true;
3784 }
3785 LogRel(("File system of '%s' is %s\n", utfFile.c_str(), RTFsTypeName(enmFsTypeFile)));
3786 LONG64 i64Size;
3787 hrc = pMedium->COMGETTER(LogicalSize)(&i64Size); H();
3788#ifdef RT_OS_WINDOWS
3789 if ( enmFsTypeFile == RTFSTYPE_FAT
3790 && i64Size >= _4G)
3791 {
3792 const char *pszUnit;
3793 uint64_t u64Print = formatDiskSize((uint64_t)i64Size, &pszUnit);
3794 i_setVMRuntimeErrorCallbackF(0, "FatPartitionDetected",
3795 N_("The medium '%ls' has a logical size of %RU64%s "
3796 "but the file system the medium is located on seems "
3797 "to be FAT(32) which cannot handle files bigger than 4GB.\n"
3798 "We strongly recommend to put all your virtual disk images and "
3799 "the snapshot folder onto an NTFS partition"),
3800 strFile.raw(), u64Print, pszUnit);
3801 }
3802#else /* !RT_OS_WINDOWS */
3803 if ( enmFsTypeFile == RTFSTYPE_FAT
3804 || enmFsTypeFile == RTFSTYPE_EXT
3805 || enmFsTypeFile == RTFSTYPE_EXT2
3806 || enmFsTypeFile == RTFSTYPE_EXT3
3807 || enmFsTypeFile == RTFSTYPE_EXT4)
3808 {
3809 RTFILE file;
3810 rc = RTFileOpen(&file, utfFile.c_str(), RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
3811 if (RT_SUCCESS(rc))
3812 {
3813 RTFOFF maxSize;
3814 /* Careful: This function will work only on selected local file systems! */
3815 rc = RTFileGetMaxSizeEx(file, &maxSize);
3816 RTFileClose(file);
3817 if ( RT_SUCCESS(rc)
3818 && maxSize > 0
3819 && i64Size > (LONG64)maxSize)
3820 {
3821 const char *pszUnitSiz;
3822 const char *pszUnitMax;
3823 uint64_t u64PrintSiz = formatDiskSize((LONG64)i64Size, &pszUnitSiz);
3824 uint64_t u64PrintMax = formatDiskSize(maxSize, &pszUnitMax);
3825 i_setVMRuntimeErrorCallbackF(0, "FatPartitionDetected", /* <= not exact but ... */
3826 N_("The medium '%ls' has a logical size of %RU64%s "
3827 "but the file system the medium is located on can "
3828 "only handle files up to %RU64%s in theory.\n"
3829 "We strongly recommend to put all your virtual disk "
3830 "images and the snapshot folder onto a proper "
3831 "file system (e.g. ext3) with a sufficient size"),
3832 strFile.raw(), u64PrintSiz, pszUnitSiz, u64PrintMax, pszUnitMax);
3833 }
3834 }
3835 }
3836#endif /* !RT_OS_WINDOWS */
3837
3838 /*
3839 * Snapshot folder:
3840 * Here we test only for a FAT partition as we had to create a dummy file otherwise
3841 */
3842 if ( enmFsTypeSnap == RTFSTYPE_FAT
3843 && i64Size >= _4G
3844 && !mfSnapshotFolderSizeWarningShown)
3845 {
3846 const char *pszUnit;
3847 uint64_t u64Print = formatDiskSize(i64Size, &pszUnit);
3848 i_setVMRuntimeErrorCallbackF(0, "FatPartitionDetected",
3849#ifdef RT_OS_WINDOWS
3850 N_("The snapshot folder of this VM '%ls' seems to be located on "
3851 "a FAT(32) file system. The logical size of the medium '%ls' "
3852 "(%RU64%s) is bigger than the maximum file size this file "
3853 "system can handle (4GB).\n"
3854 "We strongly recommend to put all your virtual disk images and "
3855 "the snapshot folder onto an NTFS partition"),
3856#else
3857 N_("The snapshot folder of this VM '%ls' seems to be located on "
3858 "a FAT(32) file system. The logical size of the medium '%ls' "
3859 "(%RU64%s) is bigger than the maximum file size this file "
3860 "system can handle (4GB).\n"
3861 "We strongly recommend to put all your virtual disk images and "
3862 "the snapshot folder onto a proper file system (e.g. ext3)"),
3863#endif
3864 strSnap.raw(), strFile.raw(), u64Print, pszUnit);
3865 /* Show this particular warning only once */
3866 mfSnapshotFolderSizeWarningShown = true;
3867 }
3868
3869#ifdef RT_OS_LINUX
3870 /*
3871 * Ext4 bug: Check if the host I/O cache is disabled and the disk image is located
3872 * on an ext4 partition. Later we have to check the Linux kernel version!
3873 * This bug apparently applies to the XFS file system as well.
3874 * Linux 2.6.36 is known to be fixed (tested with 2.6.36-rc4).
3875 */
3876
3877 char szOsRelease[128];
3878 rc = RTSystemQueryOSInfo(RTSYSOSINFO_RELEASE, szOsRelease, sizeof(szOsRelease));
3879 bool fKernelHasODirectBug = RT_FAILURE(rc)
3880 || (RTStrVersionCompare(szOsRelease, "2.6.36-rc4") < 0);
3881
3882 if ( (uCaps & MediumFormatCapabilities_Asynchronous)
3883 && !fUseHostIOCache
3884 && fKernelHasODirectBug)
3885 {
3886 if ( enmFsTypeFile == RTFSTYPE_EXT4
3887 || enmFsTypeFile == RTFSTYPE_XFS)
3888 {
3889 i_setVMRuntimeErrorCallbackF(0, "Ext4PartitionDetected",
3890 N_("The host I/O cache for at least one controller is disabled "
3891 "and the medium '%ls' for this VM "
3892 "is located on an %s partition. There is a known Linux "
3893 "kernel bug which can lead to the corruption of the virtual "
3894 "disk image under these conditions.\n"
3895 "Either enable the host I/O cache permanently in the VM "
3896 "settings or put the disk image and the snapshot folder "
3897 "onto a different file system.\n"
3898 "The host I/O cache will now be enabled for this medium"),
3899 strFile.raw(), enmFsTypeFile == RTFSTYPE_EXT4 ? "ext4" : "xfs");
3900 fUseHostIOCache = true;
3901 }
3902 else if ( ( enmFsTypeSnap == RTFSTYPE_EXT4
3903 || enmFsTypeSnap == RTFSTYPE_XFS)
3904 && !mfSnapshotFolderExt4WarningShown)
3905 {
3906 i_setVMRuntimeErrorCallbackF(0, "Ext4PartitionDetected",
3907 N_("The host I/O cache for at least one controller is disabled "
3908 "and the snapshot folder for this VM "
3909 "is located on an %s partition. There is a known Linux "
3910 "kernel bug which can lead to the corruption of the virtual "
3911 "disk image under these conditions.\n"
3912 "Either enable the host I/O cache permanently in the VM "
3913 "settings or put the disk image and the snapshot folder "
3914 "onto a different file system.\n"
3915 "The host I/O cache will now be enabled for this medium"),
3916 enmFsTypeSnap == RTFSTYPE_EXT4 ? "ext4" : "xfs");
3917 fUseHostIOCache = true;
3918 mfSnapshotFolderExt4WarningShown = true;
3919 }
3920 }
3921#endif
3922 }
3923 }
3924
3925 if (pMedium)
3926 {
3927 BOOL fHostDrive;
3928 hrc = pMedium->COMGETTER(HostDrive)(&fHostDrive); H();
3929 if ( ( lType == DeviceType_DVD
3930 || lType == DeviceType_Floppy)
3931 && !fHostDrive)
3932 {
3933 /*
3934 * Informative logging.
3935 */
3936 Bstr strFile;
3937 hrc = pMedium->COMGETTER(Location)(strFile.asOutParam()); H();
3938 Utf8Str utfFile = Utf8Str(strFile);
3939 RTFSTYPE enmFsTypeFile = RTFSTYPE_UNKNOWN;
3940 (void)RTFsQueryType(utfFile.c_str(), &enmFsTypeFile);
3941 LogRel(("File system of '%s' (%s) is %s\n",
3942 utfFile.c_str(), lType == DeviceType_DVD ? "DVD" : "Floppy",
3943 RTFsTypeName(enmFsTypeFile)));
3944 }
3945 }
3946
3947 BOOL fPassthrough;
3948 hrc = pMediumAtt->COMGETTER(Passthrough)(&fPassthrough); H();
3949
3950 ComObjPtr<IBandwidthGroup> pBwGroup;
3951 Bstr strBwGroup;
3952 hrc = pMediumAtt->COMGETTER(BandwidthGroup)(pBwGroup.asOutParam()); H();
3953
3954 if (!pBwGroup.isNull())
3955 {
3956 hrc = pBwGroup->COMGETTER(Name)(strBwGroup.asOutParam()); H();
3957 }
3958
3959 rc = i_configMedium(pLunL0,
3960 !!fPassthrough,
3961 lType,
3962 fUseHostIOCache,
3963 fBuiltinIOCache,
3964 fSetupMerge,
3965 uMergeSource,
3966 uMergeTarget,
3967 strBwGroup.isEmpty() ? NULL : Utf8Str(strBwGroup).c_str(),
3968 !!fDiscard,
3969 pMedium,
3970 aMachineState,
3971 phrc);
3972 if (RT_FAILURE(rc))
3973 return rc;
3974
3975 if (fAttachDetach)
3976 {
3977 /* Attach the new driver. */
3978 if (enmBus == StorageBus_USB)
3979 {
3980 if (fHotplug)
3981 {
3982 USBStorageDevice UsbMsd = USBStorageDevice();
3983 RTUuidCreate(&UsbMsd.mUuid);
3984 UsbMsd.iPort = uInstance;
3985 rc = PDMR3UsbCreateEmulatedDevice(pUVM, pcszDevice, pCtlInst, &UsbMsd.mUuid, NULL);
3986 if (RT_SUCCESS(rc))
3987 mUSBStorageDevices.push_back(UsbMsd);
3988 }
3989 else
3990 rc = PDMR3UsbDriverAttach(pUVM, pcszDevice, uInstance, uLUN,
3991 fHotplug ? 0 : PDM_TACH_FLAGS_NOT_HOT_PLUG, NULL /*ppBase*/);
3992 }
3993 else
3994 rc = PDMR3DeviceAttach(pUVM, pcszDevice, uInstance, uLUN,
3995 fHotplug ? 0 : PDM_TACH_FLAGS_NOT_HOT_PLUG, NULL /*ppBase*/);
3996 AssertRCReturn(rc, rc);
3997
3998 /*
3999 * Make the secret key helper interface known to the VD driver if it is attached,
4000 * so we can get notified about missing keys.
4001 */
4002 PPDMIBASE pIBase = NULL;
4003 rc = PDMR3QueryDriverOnLun(pUVM, pcszDevice, uInstance, uLUN, "VD", &pIBase);
4004 if (RT_SUCCESS(rc) && pIBase)
4005 {
4006 PPDMIMEDIA pIMedium = (PPDMIMEDIA)pIBase->pfnQueryInterface(pIBase, PDMIMEDIA_IID);
4007 if (pIMedium)
4008 {
4009 rc = pIMedium->pfnSetSecKeyIf(pIMedium, mpIfSecKey, mpIfSecKeyHlp);
4010 Assert(RT_SUCCESS(rc) || rc == VERR_NOT_SUPPORTED);
4011 }
4012 }
4013
4014 /* There is no need to handle removable medium mounting, as we
4015 * unconditionally replace everthing including the block driver level.
4016 * This means the new medium will be picked up automatically. */
4017 }
4018
4019 if (paLedDevType)
4020 paLedDevType[uLUN] = lType;
4021
4022 /* Dump the changed LUN if possible, dump the complete device otherwise */
4023 if ( aMachineState != MachineState_Starting
4024 && aMachineState != MachineState_Restoring)
4025 CFGMR3Dump(pLunL0 ? pLunL0 : pCtlInst);
4026 }
4027 catch (ConfigError &x)
4028 {
4029 // InsertConfig threw something:
4030 return x.m_vrc;
4031 }
4032
4033#undef H
4034
4035 return VINF_SUCCESS;
4036}
4037
4038int Console::i_configMedium(PCFGMNODE pLunL0,
4039 bool fPassthrough,
4040 DeviceType_T enmType,
4041 bool fUseHostIOCache,
4042 bool fBuiltinIOCache,
4043 bool fSetupMerge,
4044 unsigned uMergeSource,
4045 unsigned uMergeTarget,
4046 const char *pcszBwGroup,
4047 bool fDiscard,
4048 IMedium *pMedium,
4049 MachineState_T aMachineState,
4050 HRESULT *phrc)
4051{
4052 // InsertConfig* throws
4053 try
4054 {
4055 int rc = VINF_SUCCESS;
4056 HRESULT hrc;
4057 Bstr bstr;
4058 PCFGMNODE pLunL1 = NULL;
4059 PCFGMNODE pCfg = NULL;
4060
4061#define H() \
4062 AssertMsgReturnStmt(SUCCEEDED(hrc), ("hrc=%Rhrc\n", hrc), if (phrc) *phrc = hrc, Global::vboxStatusCodeFromCOM(hrc))
4063
4064
4065 BOOL fHostDrive = FALSE;
4066 MediumType_T mediumType = MediumType_Normal;
4067 if (pMedium)
4068 {
4069 hrc = pMedium->COMGETTER(HostDrive)(&fHostDrive); H();
4070 hrc = pMedium->COMGETTER(Type)(&mediumType); H();
4071 }
4072
4073 if (fHostDrive)
4074 {
4075 Assert(pMedium);
4076 if (enmType == DeviceType_DVD)
4077 {
4078 InsertConfigString(pLunL0, "Driver", "HostDVD");
4079 InsertConfigNode(pLunL0, "Config", &pCfg);
4080
4081 hrc = pMedium->COMGETTER(Location)(bstr.asOutParam()); H();
4082 InsertConfigString(pCfg, "Path", bstr);
4083
4084 InsertConfigInteger(pCfg, "Passthrough", fPassthrough);
4085 }
4086 else if (enmType == DeviceType_Floppy)
4087 {
4088 InsertConfigString(pLunL0, "Driver", "HostFloppy");
4089 InsertConfigNode(pLunL0, "Config", &pCfg);
4090
4091 hrc = pMedium->COMGETTER(Location)(bstr.asOutParam()); H();
4092 InsertConfigString(pCfg, "Path", bstr);
4093 }
4094 }
4095 else
4096 {
4097 InsertConfigString(pLunL0, "Driver", "Block");
4098 InsertConfigNode(pLunL0, "Config", &pCfg);
4099 switch (enmType)
4100 {
4101 case DeviceType_DVD:
4102 InsertConfigString(pCfg, "Type", "DVD");
4103 InsertConfigInteger(pCfg, "Mountable", 1);
4104 break;
4105 case DeviceType_Floppy:
4106 InsertConfigString(pCfg, "Type", "Floppy 1.44");
4107 InsertConfigInteger(pCfg, "Mountable", 1);
4108 break;
4109 case DeviceType_HardDisk:
4110 default:
4111 InsertConfigString(pCfg, "Type", "HardDisk");
4112 InsertConfigInteger(pCfg, "Mountable", 0);
4113 }
4114
4115 if ( pMedium
4116 && ( enmType == DeviceType_DVD
4117 || enmType == DeviceType_Floppy)
4118 )
4119 {
4120 // if this medium represents an ISO image and this image is inaccessible,
4121 // the ignore it instead of causing a failure; this can happen when we
4122 // restore a VM state and the ISO has disappeared, e.g. because the Guest
4123 // Additions were mounted and the user upgraded VirtualBox. Previously
4124 // we failed on startup, but that's not good because the only way out then
4125 // would be to discard the VM state...
4126 MediumState_T mediumState;
4127 hrc = pMedium->RefreshState(&mediumState); H();
4128 if (mediumState == MediumState_Inaccessible)
4129 {
4130 Bstr loc;
4131 hrc = pMedium->COMGETTER(Location)(loc.asOutParam()); H();
4132 i_setVMRuntimeErrorCallbackF(0, "DvdOrFloppyImageInaccessible",
4133 "The image file '%ls' is inaccessible and is being ignored. "
4134 "Please select a different image file for the virtual %s drive.",
4135 loc.raw(),
4136 enmType == DeviceType_DVD ? "DVD" : "floppy");
4137 pMedium = NULL;
4138 }
4139 }
4140
4141 if (pMedium)
4142 {
4143 /* Start with length of parent chain, as the list is reversed */
4144 unsigned uImage = 0;
4145 IMedium *pTmp = pMedium;
4146 while (pTmp)
4147 {
4148 uImage++;
4149 hrc = pTmp->COMGETTER(Parent)(&pTmp); H();
4150 }
4151 /* Index of last image */
4152 uImage--;
4153
4154#if 0 /* Enable for I/O debugging */
4155 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL0);
4156 InsertConfigString(pLunL0, "Driver", "DiskIntegrity");
4157 InsertConfigNode(pLunL0, "Config", &pCfg);
4158 InsertConfigInteger(pCfg, "CheckConsistency", 0);
4159 InsertConfigInteger(pCfg, "CheckDoubleCompletions", 1);
4160#endif
4161
4162 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL1);
4163 InsertConfigString(pLunL1, "Driver", "VD");
4164 InsertConfigNode(pLunL1, "Config", &pCfg);
4165
4166# ifdef VBOX_WITH_EXTPACK
4167 static const Utf8Str strExtPackPuel("Oracle VM VirtualBox Extension Pack");
4168 static const char *s_pszVDPlugin = "VDPluginCrypt";
4169 if (mptrExtPackManager->i_isExtPackUsable(strExtPackPuel.c_str()))
4170 {
4171 /* Configure loading the VDPlugin. */
4172 PCFGMNODE pCfgPlugins = NULL;
4173 PCFGMNODE pCfgPlugin = NULL;
4174 Utf8Str strPlugin;
4175 hrc = mptrExtPackManager->i_getLibraryPathForExtPack(s_pszVDPlugin, &strExtPackPuel, &strPlugin);
4176 // Don't fail, this is optional!
4177 if (SUCCEEDED(hrc))
4178 {
4179 InsertConfigNode(pCfg, "Plugins", &pCfgPlugins);
4180 InsertConfigNode(pCfgPlugins, s_pszVDPlugin, &pCfgPlugin);
4181 InsertConfigString(pCfgPlugin, "Path", strPlugin.c_str());
4182 }
4183 }
4184# endif
4185
4186 hrc = pMedium->COMGETTER(Location)(bstr.asOutParam()); H();
4187 InsertConfigString(pCfg, "Path", bstr);
4188
4189 hrc = pMedium->COMGETTER(Format)(bstr.asOutParam()); H();
4190 InsertConfigString(pCfg, "Format", bstr);
4191
4192 if (mediumType == MediumType_Readonly)
4193 InsertConfigInteger(pCfg, "ReadOnly", 1);
4194 else if (enmType == DeviceType_Floppy)
4195 InsertConfigInteger(pCfg, "MaybeReadOnly", 1);
4196
4197 /* Start without exclusive write access to the images. */
4198 /** @todo Live Migration: I don't quite like this, we risk screwing up when
4199 * we're resuming the VM if some 3rd dude have any of the VDIs open
4200 * with write sharing denied. However, if the two VMs are sharing a
4201 * image it really is necessary....
4202 *
4203 * So, on the "lock-media" command, the target teleporter should also
4204 * make DrvVD undo TempReadOnly. It gets interesting if we fail after
4205 * that. Grumble. */
4206 if ( enmType == DeviceType_HardDisk
4207 && ( aMachineState == MachineState_TeleportingIn
4208 || aMachineState == MachineState_FaultTolerantSyncing))
4209 InsertConfigInteger(pCfg, "TempReadOnly", 1);
4210
4211 /* Flag for opening the medium for sharing between VMs. This
4212 * is done at the moment only for the first (and only) medium
4213 * in the chain, as shared media can have no diffs. */
4214 if (mediumType == MediumType_Shareable)
4215 InsertConfigInteger(pCfg, "Shareable", 1);
4216
4217 if (!fUseHostIOCache)
4218 {
4219 InsertConfigInteger(pCfg, "UseNewIo", 1);
4220 /*
4221 * Activate the builtin I/O cache for harddisks only.
4222 * It caches writes only which doesn't make sense for DVD drives
4223 * and just increases the overhead.
4224 */
4225 if ( fBuiltinIOCache
4226 && (enmType == DeviceType_HardDisk))
4227 InsertConfigInteger(pCfg, "BlockCache", 1);
4228 }
4229
4230 if (fSetupMerge)
4231 {
4232 InsertConfigInteger(pCfg, "SetupMerge", 1);
4233 if (uImage == uMergeSource)
4234 InsertConfigInteger(pCfg, "MergeSource", 1);
4235 else if (uImage == uMergeTarget)
4236 InsertConfigInteger(pCfg, "MergeTarget", 1);
4237 }
4238
4239 switch (enmType)
4240 {
4241 case DeviceType_DVD:
4242 InsertConfigString(pCfg, "Type", "DVD");
4243 break;
4244 case DeviceType_Floppy:
4245 InsertConfigString(pCfg, "Type", "Floppy");
4246 break;
4247 case DeviceType_HardDisk:
4248 default:
4249 InsertConfigString(pCfg, "Type", "HardDisk");
4250 }
4251
4252 if (pcszBwGroup)
4253 InsertConfigString(pCfg, "BwGroup", pcszBwGroup);
4254
4255 if (fDiscard)
4256 InsertConfigInteger(pCfg, "Discard", 1);
4257
4258 /* Pass all custom parameters. */
4259 bool fHostIP = true;
4260 bool fEncrypted = false;
4261 hrc = i_configMediumProperties(pCfg, pMedium, &fHostIP, &fEncrypted); H();
4262
4263 /* Create an inverted list of parents. */
4264 uImage--;
4265 IMedium *pParentMedium = pMedium;
4266 for (PCFGMNODE pParent = pCfg;; uImage--)
4267 {
4268 hrc = pParentMedium->COMGETTER(Parent)(&pMedium); H();
4269 if (!pMedium)
4270 break;
4271
4272 PCFGMNODE pCur;
4273 InsertConfigNode(pParent, "Parent", &pCur);
4274 hrc = pMedium->COMGETTER(Location)(bstr.asOutParam()); H();
4275 InsertConfigString(pCur, "Path", bstr);
4276
4277 hrc = pMedium->COMGETTER(Format)(bstr.asOutParam()); H();
4278 InsertConfigString(pCur, "Format", bstr);
4279
4280 if (fSetupMerge)
4281 {
4282 if (uImage == uMergeSource)
4283 InsertConfigInteger(pCur, "MergeSource", 1);
4284 else if (uImage == uMergeTarget)
4285 InsertConfigInteger(pCur, "MergeTarget", 1);
4286 }
4287
4288 /* Configure medium properties. */
4289 hrc = i_configMediumProperties(pCur, pMedium, &fHostIP, &fEncrypted); H();
4290
4291 /* next */
4292 pParent = pCur;
4293 pParentMedium = pMedium;
4294 }
4295
4296 /* Custom code: put marker to not use host IP stack to driver
4297 * configuration node. Simplifies life of DrvVD a bit. */
4298 if (!fHostIP)
4299 InsertConfigInteger(pCfg, "HostIPStack", 0);
4300
4301 if (fEncrypted)
4302 m_cDisksEncrypted++;
4303 }
4304 }
4305#undef H
4306 }
4307 catch (ConfigError &x)
4308 {
4309 // InsertConfig threw something:
4310 return x.m_vrc;
4311 }
4312
4313 return VINF_SUCCESS;
4314}
4315
4316/**
4317 * Adds the medium properties to the CFGM tree.
4318 *
4319 * @returns VBox status code.
4320 * @param pCur The current CFGM node.
4321 * @param pMedium The medium object to configure.
4322 * @param pfHostIP Where to return the value of the \"HostIPStack\" property if found.
4323 * @param pfEncrypted Where to return whether the medium is encrypted.
4324 */
4325int Console::i_configMediumProperties(PCFGMNODE pCur, IMedium *pMedium, bool *pfHostIP, bool *pfEncrypted)
4326{
4327 /* Pass all custom parameters. */
4328 SafeArray<BSTR> aNames;
4329 SafeArray<BSTR> aValues;
4330 HRESULT hrc = pMedium->GetProperties(NULL, ComSafeArrayAsOutParam(aNames),
4331 ComSafeArrayAsOutParam(aValues));
4332
4333 if ( SUCCEEDED(hrc)
4334 && aNames.size() != 0)
4335 {
4336 PCFGMNODE pVDC;
4337 InsertConfigNode(pCur, "VDConfig", &pVDC);
4338 for (size_t ii = 0; ii < aNames.size(); ++ii)
4339 {
4340 if (aValues[ii] && *aValues[ii])
4341 {
4342 Utf8Str name = aNames[ii];
4343 Utf8Str value = aValues[ii];
4344 size_t offSlash = name.find("/", 0);
4345 if ( offSlash != name.npos
4346 && !name.startsWith("Special/"))
4347 {
4348 com::Utf8Str strFilter;
4349 com::Utf8Str strKey;
4350
4351 hrc = strFilter.assignEx(name, 0, offSlash);
4352 if (FAILED(hrc))
4353 break;
4354
4355 hrc = strKey.assignEx(name, offSlash + 1, name.length() - offSlash - 1); /* Skip slash */
4356 if (FAILED(hrc))
4357 break;
4358
4359 PCFGMNODE pCfgFilterConfig = CFGMR3GetChild(pVDC, strFilter.c_str());
4360 if (!pCfgFilterConfig)
4361 InsertConfigNode(pVDC, strFilter.c_str(), &pCfgFilterConfig);
4362
4363 InsertConfigString(pCfgFilterConfig, strKey.c_str(), value);
4364 }
4365 else
4366 {
4367 InsertConfigString(pVDC, name.c_str(), value);
4368 if ( name.compare("HostIPStack") == 0
4369 && value.compare("0") == 0)
4370 *pfHostIP = false;
4371 }
4372
4373 if ( name.compare("CRYPT/KeyId") == 0
4374 && pfEncrypted)
4375 *pfEncrypted = true;
4376 }
4377 }
4378 }
4379
4380 return hrc;
4381}
4382
4383/**
4384 * Construct the Network configuration tree
4385 *
4386 * @returns VBox status code.
4387 *
4388 * @param pszDevice The PDM device name.
4389 * @param uInstance The PDM device instance.
4390 * @param uLun The PDM LUN number of the drive.
4391 * @param aNetworkAdapter The network adapter whose attachment needs to be changed
4392 * @param pCfg Configuration node for the device
4393 * @param pLunL0 To store the pointer to the LUN#0.
4394 * @param pInst The instance CFGM node
4395 * @param fAttachDetach To determine if the network attachment should
4396 * be attached/detached after/before
4397 * configuration.
4398 * @param fIgnoreConnectFailure
4399 * True if connection failures should be ignored
4400 * (makes only sense for bridged/host-only networks).
4401 *
4402 * @note Locks this object for writing.
4403 * @thread EMT
4404 */
4405int Console::i_configNetwork(const char *pszDevice,
4406 unsigned uInstance,
4407 unsigned uLun,
4408 INetworkAdapter *aNetworkAdapter,
4409 PCFGMNODE pCfg,
4410 PCFGMNODE pLunL0,
4411 PCFGMNODE pInst,
4412 bool fAttachDetach,
4413 bool fIgnoreConnectFailure)
4414{
4415 AutoCaller autoCaller(this);
4416 AssertComRCReturn(autoCaller.rc(), VERR_ACCESS_DENIED);
4417
4418 // InsertConfig* throws
4419 try
4420 {
4421 int rc = VINF_SUCCESS;
4422 HRESULT hrc;
4423 Bstr bstr;
4424
4425#define H() AssertLogRelMsgReturn(!FAILED(hrc), ("hrc=%Rhrc\n", hrc), VERR_MAIN_CONFIG_CONSTRUCTOR_COM_ERROR)
4426
4427 /*
4428 * Locking the object before doing VMR3* calls is quite safe here, since
4429 * we're on EMT. Write lock is necessary because we indirectly modify the
4430 * meAttachmentType member.
4431 */
4432 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
4433
4434 ComPtr<IMachine> pMachine = i_machine();
4435
4436 ComPtr<IVirtualBox> virtualBox;
4437 hrc = pMachine->COMGETTER(Parent)(virtualBox.asOutParam()); H();
4438
4439 ComPtr<IHost> host;
4440 hrc = virtualBox->COMGETTER(Host)(host.asOutParam()); H();
4441
4442 BOOL fSniffer;
4443 hrc = aNetworkAdapter->COMGETTER(TraceEnabled)(&fSniffer); H();
4444
4445 NetworkAdapterPromiscModePolicy_T enmPromiscModePolicy;
4446 hrc = aNetworkAdapter->COMGETTER(PromiscModePolicy)(&enmPromiscModePolicy); H();
4447 const char *pszPromiscuousGuestPolicy;
4448 switch (enmPromiscModePolicy)
4449 {
4450 case NetworkAdapterPromiscModePolicy_Deny: pszPromiscuousGuestPolicy = "deny"; break;
4451 case NetworkAdapterPromiscModePolicy_AllowNetwork: pszPromiscuousGuestPolicy = "allow-network"; break;
4452 case NetworkAdapterPromiscModePolicy_AllowAll: pszPromiscuousGuestPolicy = "allow-all"; break;
4453 default: AssertFailedReturn(VERR_INTERNAL_ERROR_4);
4454 }
4455
4456 if (fAttachDetach)
4457 {
4458 rc = PDMR3DeviceDetach(mpUVM, pszDevice, uInstance, uLun, 0 /*fFlags*/);
4459 if (rc == VINF_PDM_NO_DRIVER_ATTACHED_TO_LUN)
4460 rc = VINF_SUCCESS;
4461 AssertLogRelRCReturn(rc, rc);
4462
4463 /* nuke anything which might have been left behind. */
4464 CFGMR3RemoveNode(CFGMR3GetChildF(pInst, "LUN#%u", uLun));
4465 }
4466
4467#ifdef VBOX_WITH_NETSHAPER
4468 ComObjPtr<IBandwidthGroup> pBwGroup;
4469 Bstr strBwGroup;
4470 hrc = aNetworkAdapter->COMGETTER(BandwidthGroup)(pBwGroup.asOutParam()); H();
4471
4472 if (!pBwGroup.isNull())
4473 {
4474 hrc = pBwGroup->COMGETTER(Name)(strBwGroup.asOutParam()); H();
4475 }
4476#endif /* VBOX_WITH_NETSHAPER */
4477
4478 Utf8Str strNetDriver;
4479
4480
4481 InsertConfigNode(pInst, "LUN#0", &pLunL0);
4482
4483#ifdef VBOX_WITH_NETSHAPER
4484 if (!strBwGroup.isEmpty())
4485 {
4486 InsertConfigString(pLunL0, "Driver", "NetShaper");
4487 InsertConfigNode(pLunL0, "Config", &pCfg);
4488 InsertConfigString(pCfg, "BwGroup", strBwGroup);
4489 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL0);
4490 }
4491#endif /* VBOX_WITH_NETSHAPER */
4492
4493 if (fSniffer)
4494 {
4495 InsertConfigString(pLunL0, "Driver", "NetSniffer");
4496 InsertConfigNode(pLunL0, "Config", &pCfg);
4497 hrc = aNetworkAdapter->COMGETTER(TraceFile)(bstr.asOutParam()); H();
4498 if (!bstr.isEmpty()) /* check convention for indicating default file. */
4499 InsertConfigString(pCfg, "File", bstr);
4500 InsertConfigNode(pLunL0, "AttachedDriver", &pLunL0);
4501 }
4502
4503
4504 Bstr networkName, trunkName, trunkType;
4505 NetworkAttachmentType_T eAttachmentType;
4506 hrc = aNetworkAdapter->COMGETTER(AttachmentType)(&eAttachmentType); H();
4507 switch (eAttachmentType)
4508 {
4509 case NetworkAttachmentType_Null:
4510 break;
4511
4512 case NetworkAttachmentType_NAT:
4513 {
4514 ComPtr<INATEngine> natEngine;
4515 hrc = aNetworkAdapter->COMGETTER(NATEngine)(natEngine.asOutParam()); H();
4516 InsertConfigString(pLunL0, "Driver", "NAT");
4517 InsertConfigNode(pLunL0, "Config", &pCfg);
4518
4519 /* Configure TFTP prefix and boot filename. */
4520 hrc = virtualBox->COMGETTER(HomeFolder)(bstr.asOutParam()); H();
4521 if (!bstr.isEmpty())
4522 InsertConfigString(pCfg, "TFTPPrefix", Utf8StrFmt("%ls%c%s", bstr.raw(), RTPATH_DELIMITER, "TFTP"));
4523 hrc = pMachine->COMGETTER(Name)(bstr.asOutParam()); H();
4524 InsertConfigString(pCfg, "BootFile", Utf8StrFmt("%ls.pxe", bstr.raw()));
4525
4526 hrc = natEngine->COMGETTER(Network)(bstr.asOutParam()); H();
4527 if (!bstr.isEmpty())
4528 InsertConfigString(pCfg, "Network", bstr);
4529 else
4530 {
4531 ULONG uSlot;
4532 hrc = aNetworkAdapter->COMGETTER(Slot)(&uSlot); H();
4533 InsertConfigString(pCfg, "Network", Utf8StrFmt("10.0.%d.0/24", uSlot+2));
4534 }
4535 hrc = natEngine->COMGETTER(HostIP)(bstr.asOutParam()); H();
4536 if (!bstr.isEmpty())
4537 InsertConfigString(pCfg, "BindIP", bstr);
4538 ULONG mtu = 0;
4539 ULONG sockSnd = 0;
4540 ULONG sockRcv = 0;
4541 ULONG tcpSnd = 0;
4542 ULONG tcpRcv = 0;
4543 hrc = natEngine->GetNetworkSettings(&mtu, &sockSnd, &sockRcv, &tcpSnd, &tcpRcv); H();
4544 if (mtu)
4545 InsertConfigInteger(pCfg, "SlirpMTU", mtu);
4546 if (sockRcv)
4547 InsertConfigInteger(pCfg, "SockRcv", sockRcv);
4548 if (sockSnd)
4549 InsertConfigInteger(pCfg, "SockSnd", sockSnd);
4550 if (tcpRcv)
4551 InsertConfigInteger(pCfg, "TcpRcv", tcpRcv);
4552 if (tcpSnd)
4553 InsertConfigInteger(pCfg, "TcpSnd", tcpSnd);
4554 hrc = natEngine->COMGETTER(TFTPPrefix)(bstr.asOutParam()); H();
4555 if (!bstr.isEmpty())
4556 {
4557 RemoveConfigValue(pCfg, "TFTPPrefix");
4558 InsertConfigString(pCfg, "TFTPPrefix", bstr);
4559 }
4560 hrc = natEngine->COMGETTER(TFTPBootFile)(bstr.asOutParam()); H();
4561 if (!bstr.isEmpty())
4562 {
4563 RemoveConfigValue(pCfg, "BootFile");
4564 InsertConfigString(pCfg, "BootFile", bstr);
4565 }
4566 hrc = natEngine->COMGETTER(TFTPNextServer)(bstr.asOutParam()); H();
4567 if (!bstr.isEmpty())
4568 InsertConfigString(pCfg, "NextServer", bstr);
4569 BOOL fDNSFlag;
4570 hrc = natEngine->COMGETTER(DNSPassDomain)(&fDNSFlag); H();
4571 InsertConfigInteger(pCfg, "PassDomain", fDNSFlag);
4572 hrc = natEngine->COMGETTER(DNSProxy)(&fDNSFlag); H();
4573 InsertConfigInteger(pCfg, "DNSProxy", fDNSFlag);
4574 hrc = natEngine->COMGETTER(DNSUseHostResolver)(&fDNSFlag); H();
4575 InsertConfigInteger(pCfg, "UseHostResolver", fDNSFlag);
4576
4577 ULONG aliasMode;
4578 hrc = natEngine->COMGETTER(AliasMode)(&aliasMode); H();
4579 InsertConfigInteger(pCfg, "AliasMode", aliasMode);
4580
4581 /* port-forwarding */
4582 SafeArray<BSTR> pfs;
4583 hrc = natEngine->COMGETTER(Redirects)(ComSafeArrayAsOutParam(pfs)); H();
4584 PCFGMNODE pPF = NULL; /* /Devices/Dev/.../Config/PF#0/ */
4585 for (unsigned int i = 0; i < pfs.size(); ++i)
4586 {
4587 uint16_t port = 0;
4588 BSTR r = pfs[i];
4589 Utf8Str utf = Utf8Str(r);
4590 Utf8Str strName;
4591 Utf8Str strProto;
4592 Utf8Str strHostPort;
4593 Utf8Str strHostIP;
4594 Utf8Str strGuestPort;
4595 Utf8Str strGuestIP;
4596 size_t pos, ppos;
4597 pos = ppos = 0;
4598#define ITERATE_TO_NEXT_TERM(res, str, pos, ppos) \
4599 do { \
4600 pos = str.find(",", ppos); \
4601 if (pos == Utf8Str::npos) \
4602 { \
4603 Log(( #res " extracting from %s is failed\n", str.c_str())); \
4604 continue; \
4605 } \
4606 res = str.substr(ppos, pos - ppos); \
4607 Log2((#res " %s pos:%d, ppos:%d\n", res.c_str(), pos, ppos)); \
4608 ppos = pos + 1; \
4609 } while (0)
4610 ITERATE_TO_NEXT_TERM(strName, utf, pos, ppos);
4611 ITERATE_TO_NEXT_TERM(strProto, utf, pos, ppos);
4612 ITERATE_TO_NEXT_TERM(strHostIP, utf, pos, ppos);
4613 ITERATE_TO_NEXT_TERM(strHostPort, utf, pos, ppos);
4614 ITERATE_TO_NEXT_TERM(strGuestIP, utf, pos, ppos);
4615 strGuestPort = utf.substr(ppos, utf.length() - ppos);
4616#undef ITERATE_TO_NEXT_TERM
4617
4618 uint32_t proto = strProto.toUInt32();
4619 bool fValid = true;
4620 switch (proto)
4621 {
4622 case NATProtocol_UDP:
4623 strProto = "UDP";
4624 break;
4625 case NATProtocol_TCP:
4626 strProto = "TCP";
4627 break;
4628 default:
4629 fValid = false;
4630 }
4631 /* continue with next rule if no valid proto was passed */
4632 if (!fValid)
4633 continue;
4634
4635 if (strName.isEmpty())
4636 VMSetError(VMR3GetVM(mpUVM), VERR_CFGM_NO_NODE, RT_SRC_POS,
4637 N_("NAT redirection rule without a name"));
4638
4639 InsertConfigNode(pCfg, strName.c_str(), &pPF);
4640 InsertConfigString(pPF, "Protocol", strProto);
4641
4642 if (!strHostIP.isEmpty())
4643 InsertConfigString(pPF, "BindIP", strHostIP);
4644
4645 if (!strGuestIP.isEmpty())
4646 InsertConfigString(pPF, "GuestIP", strGuestIP);
4647
4648 port = RTStrToUInt16(strHostPort.c_str());
4649 if (port)
4650 InsertConfigInteger(pPF, "HostPort", port);
4651
4652 port = RTStrToUInt16(strGuestPort.c_str());
4653 if (port)
4654 InsertConfigInteger(pPF, "GuestPort", port);
4655 }
4656 break;
4657 }
4658
4659 case NetworkAttachmentType_Bridged:
4660 {
4661#if (defined(RT_OS_LINUX) || defined(RT_OS_FREEBSD)) && !defined(VBOX_WITH_NETFLT)
4662 hrc = i_attachToTapInterface(aNetworkAdapter);
4663 if (FAILED(hrc))
4664 {
4665 switch (hrc)
4666 {
4667 case VERR_ACCESS_DENIED:
4668 return VMSetError(VMR3GetVM(mpUVM), VERR_HOSTIF_INIT_FAILED, RT_SRC_POS, N_(
4669 "Failed to open '/dev/net/tun' for read/write access. Please check the "
4670 "permissions of that node. Either run 'chmod 0666 /dev/net/tun' or "
4671 "change the group of that node and make yourself a member of that group. Make "
4672 "sure that these changes are permanent, especially if you are "
4673 "using udev"));
4674 default:
4675 AssertMsgFailed(("Could not attach to host interface! Bad!\n"));
4676 return VMSetError(VMR3GetVM(mpUVM), VERR_HOSTIF_INIT_FAILED, RT_SRC_POS, N_(
4677 "Failed to initialize Host Interface Networking"));
4678 }
4679 }
4680
4681 Assert((intptr_t)maTapFD[uInstance] >= 0);
4682 if ((intptr_t)maTapFD[uInstance] >= 0)
4683 {
4684 InsertConfigString(pLunL0, "Driver", "HostInterface");
4685 InsertConfigNode(pLunL0, "Config", &pCfg);
4686 InsertConfigInteger(pCfg, "FileHandle", (intptr_t)maTapFD[uInstance]);
4687 }
4688
4689#elif defined(VBOX_WITH_NETFLT)
4690 /*
4691 * This is the new VBoxNetFlt+IntNet stuff.
4692 */
4693 Bstr BridgedIfName;
4694 hrc = aNetworkAdapter->COMGETTER(BridgedInterface)(BridgedIfName.asOutParam());
4695 if (FAILED(hrc))
4696 {
4697 LogRel(("NetworkAttachmentType_Bridged: COMGETTER(BridgedInterface) failed, hrc (0x%x)\n", hrc));
4698 H();
4699 }
4700
4701 Utf8Str BridgedIfNameUtf8(BridgedIfName);
4702 const char *pszBridgedIfName = BridgedIfNameUtf8.c_str();
4703
4704# if defined(RT_OS_DARWIN)
4705 /* The name is on the form 'ifX: long name', chop it off at the colon. */
4706 char szTrunk[8];
4707 RTStrCopy(szTrunk, sizeof(szTrunk), pszBridgedIfName);
4708 char *pszColon = (char *)memchr(szTrunk, ':', sizeof(szTrunk));
4709// Quick fix for @bugref{5633}
4710// if (!pszColon)
4711// {
4712// /*
4713// * Dynamic changing of attachment causes an attempt to configure
4714// * network with invalid host adapter (as it is must be changed before
4715// * the attachment), calling Detach here will cause a deadlock.
4716// * See @bugref{4750}.
4717// * hrc = aNetworkAdapter->Detach(); H();
4718// */
4719// return VMSetError(VMR3GetVM(mpUVM), VERR_INTERNAL_ERROR, RT_SRC_POS,
4720// N_("Malformed host interface networking name '%ls'"),
4721// BridgedIfName.raw());
4722// }
4723 if (pszColon)
4724 *pszColon = '\0';
4725 const char *pszTrunk = szTrunk;
4726
4727# elif defined(RT_OS_SOLARIS)
4728 /* The name is on the form format 'ifX[:1] - long name, chop it off at space. */
4729 char szTrunk[256];
4730 strlcpy(szTrunk, pszBridgedIfName, sizeof(szTrunk));
4731 char *pszSpace = (char *)memchr(szTrunk, ' ', sizeof(szTrunk));
4732
4733 /*
4734 * Currently don't bother about malformed names here for the sake of people using
4735 * VBoxManage and setting only the NIC name from there. If there is a space we
4736 * chop it off and proceed, otherwise just use whatever we've got.
4737 */
4738 if (pszSpace)
4739 *pszSpace = '\0';
4740
4741 /* Chop it off at the colon (zone naming eg: e1000g:1 we need only the e1000g) */
4742 char *pszColon = (char *)memchr(szTrunk, ':', sizeof(szTrunk));
4743 if (pszColon)
4744 *pszColon = '\0';
4745
4746 const char *pszTrunk = szTrunk;
4747
4748# elif defined(RT_OS_WINDOWS)
4749 ComPtr<IHostNetworkInterface> hostInterface;
4750 hrc = host->FindHostNetworkInterfaceByName(BridgedIfName.raw(),
4751 hostInterface.asOutParam());
4752 if (!SUCCEEDED(hrc))
4753 {
4754 AssertLogRelMsgFailed(("NetworkAttachmentType_Bridged: FindByName failed, rc=%Rhrc (0x%x)", hrc, hrc));
4755 return VMSetError(VMR3GetVM(mpUVM), VERR_INTERNAL_ERROR, RT_SRC_POS,
4756 N_("Nonexistent host networking interface, name '%ls'"),
4757 BridgedIfName.raw());
4758 }
4759
4760 HostNetworkInterfaceType_T eIfType;
4761 hrc = hostInterface->COMGETTER(InterfaceType)(&eIfType);
4762 if (FAILED(hrc))
4763 {
4764 LogRel(("NetworkAttachmentType_Bridged: COMGETTER(InterfaceType) failed, hrc (0x%x)\n", hrc));
4765 H();
4766 }
4767
4768 if (eIfType != HostNetworkInterfaceType_Bridged)
4769 {
4770 return VMSetError(VMR3GetVM(mpUVM), VERR_INTERNAL_ERROR, RT_SRC_POS,
4771 N_("Interface ('%ls') is not a Bridged Adapter interface"),
4772 BridgedIfName.raw());
4773 }
4774
4775 hrc = hostInterface->COMGETTER(Id)(bstr.asOutParam());
4776 if (FAILED(hrc))
4777 {
4778 LogRel(("NetworkAttachmentType_Bridged: COMGETTER(Id) failed, hrc (0x%x)\n", hrc));
4779 H();
4780 }
4781 Guid hostIFGuid(bstr);
4782
4783 INetCfg *pNc;
4784 ComPtr<INetCfgComponent> pAdaptorComponent;
4785 LPWSTR pszApp;
4786
4787 hrc = VBoxNetCfgWinQueryINetCfg(&pNc, FALSE, L"VirtualBox", 10, &pszApp);
4788 Assert(hrc == S_OK);
4789 if (hrc != S_OK)
4790 {
4791 LogRel(("NetworkAttachmentType_Bridged: Failed to get NetCfg, hrc=%Rhrc (0x%x)\n", hrc, hrc));
4792 H();
4793 }
4794
4795 /* get the adapter's INetCfgComponent*/
4796 hrc = VBoxNetCfgWinGetComponentByGuid(pNc, &GUID_DEVCLASS_NET, (GUID*)hostIFGuid.raw(),
4797 pAdaptorComponent.asOutParam());
4798 if (hrc != S_OK)
4799 {
4800 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
4801 LogRel(("NetworkAttachmentType_Bridged: VBoxNetCfgWinGetComponentByGuid failed, hrc (0x%x)\n", hrc));
4802 H();
4803 }
4804#define VBOX_WIN_BINDNAME_PREFIX "\\DEVICE\\"
4805 char szTrunkName[INTNET_MAX_TRUNK_NAME];
4806 char *pszTrunkName = szTrunkName;
4807 wchar_t * pswzBindName;
4808 hrc = pAdaptorComponent->GetBindName(&pswzBindName);
4809 Assert(hrc == S_OK);
4810 if (hrc == S_OK)
4811 {
4812 int cwBindName = (int)wcslen(pswzBindName) + 1;
4813 int cbFullBindNamePrefix = sizeof(VBOX_WIN_BINDNAME_PREFIX);
4814 if (sizeof(szTrunkName) > cbFullBindNamePrefix + cwBindName)
4815 {
4816 strcpy(szTrunkName, VBOX_WIN_BINDNAME_PREFIX);
4817 pszTrunkName += cbFullBindNamePrefix-1;
4818 if (!WideCharToMultiByte(CP_ACP, 0, pswzBindName, cwBindName, pszTrunkName,
4819 sizeof(szTrunkName) - cbFullBindNamePrefix + 1, NULL, NULL))
4820 {
4821 DWORD err = GetLastError();
4822 hrc = HRESULT_FROM_WIN32(err);
4823 AssertMsgFailed(("%hrc=%Rhrc %#x\n", hrc, hrc));
4824 AssertLogRelMsgFailed(("NetworkAttachmentType_Bridged: WideCharToMultiByte failed, hr=%Rhrc (0x%x) err=%u\n",
4825 hrc, hrc, err));
4826 }
4827 }
4828 else
4829 {
4830 AssertLogRelMsgFailed(("NetworkAttachmentType_Bridged: insufficient szTrunkName buffer space\n"));
4831 /** @todo set appropriate error code */
4832 hrc = E_FAIL;
4833 }
4834
4835 if (hrc != S_OK)
4836 {
4837 AssertFailed();
4838 CoTaskMemFree(pswzBindName);
4839 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
4840 H();
4841 }
4842
4843 /* we're not freeing the bind name since we'll use it later for detecting wireless*/
4844 }
4845 else
4846 {
4847 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
4848 AssertLogRelMsgFailed(("NetworkAttachmentType_Bridged: VBoxNetCfgWinGetComponentByGuid failed, hrc (0x%x)",
4849 hrc));
4850 H();
4851 }
4852
4853 const char *pszTrunk = szTrunkName;
4854 /* we're not releasing the INetCfg stuff here since we use it later to figure out whether it is wireless */
4855
4856# elif defined(RT_OS_LINUX) || defined(RT_OS_FREEBSD)
4857# if defined(RT_OS_FREEBSD)
4858 /*
4859 * If we bridge to a tap interface open it the `old' direct way.
4860 * This works and performs better than bridging a physical
4861 * interface via the current FreeBSD vboxnetflt implementation.
4862 */
4863 if (!strncmp(pszBridgedIfName, RT_STR_TUPLE("tap"))) {
4864 hrc = i_attachToTapInterface(aNetworkAdapter);
4865 if (FAILED(hrc))
4866 {
4867 switch (hrc)
4868 {
4869 case VERR_ACCESS_DENIED:
4870 return VMSetError(VMR3GetVM(mpUVM), VERR_HOSTIF_INIT_FAILED, RT_SRC_POS, N_(
4871 "Failed to open '/dev/%s' for read/write access. Please check the "
4872 "permissions of that node, and that the net.link.tap.user_open "
4873 "sysctl is set. Either run 'chmod 0666 /dev/%s' or "
4874 "change the group of that node to vboxusers and make yourself "
4875 "a member of that group. Make sure that these changes are permanent."),
4876 pszBridgedIfName, pszBridgedIfName);
4877 default:
4878 AssertMsgFailed(("Could not attach to tap interface! Bad!\n"));
4879 return VMSetError(VMR3GetVM(mpUVM), VERR_HOSTIF_INIT_FAILED, RT_SRC_POS, N_(
4880 "Failed to initialize Host Interface Networking"));
4881 }
4882 }
4883
4884 Assert((intptr_t)maTapFD[uInstance] >= 0);
4885 if ((intptr_t)maTapFD[uInstance] >= 0)
4886 {
4887 InsertConfigString(pLunL0, "Driver", "HostInterface");
4888 InsertConfigNode(pLunL0, "Config", &pCfg);
4889 InsertConfigInteger(pCfg, "FileHandle", (intptr_t)maTapFD[uInstance]);
4890 }
4891 break;
4892 }
4893# endif
4894 /** @todo Check for malformed names. */
4895 const char *pszTrunk = pszBridgedIfName;
4896
4897 /* Issue a warning if the interface is down */
4898 {
4899 int iSock = socket(AF_INET, SOCK_DGRAM, 0);
4900 if (iSock >= 0)
4901 {
4902 struct ifreq Req;
4903 RT_ZERO(Req);
4904 RTStrCopy(Req.ifr_name, sizeof(Req.ifr_name), pszBridgedIfName);
4905 if (ioctl(iSock, SIOCGIFFLAGS, &Req) >= 0)
4906 if ((Req.ifr_flags & IFF_UP) == 0)
4907 i_setVMRuntimeErrorCallbackF(0, "BridgedInterfaceDown",
4908 N_("Bridged interface %s is down. Guest will not be able to use this interface"),
4909 pszBridgedIfName);
4910
4911 close(iSock);
4912 }
4913 }
4914
4915# else
4916# error "PORTME (VBOX_WITH_NETFLT)"
4917# endif
4918
4919 InsertConfigString(pLunL0, "Driver", "IntNet");
4920 InsertConfigNode(pLunL0, "Config", &pCfg);
4921 InsertConfigString(pCfg, "Trunk", pszTrunk);
4922 InsertConfigInteger(pCfg, "TrunkType", kIntNetTrunkType_NetFlt);
4923 InsertConfigInteger(pCfg, "IgnoreConnectFailure", (uint64_t)fIgnoreConnectFailure);
4924 InsertConfigString(pCfg, "IfPolicyPromisc", pszPromiscuousGuestPolicy);
4925 char szNetwork[INTNET_MAX_NETWORK_NAME];
4926
4927#if defined(RT_OS_SOLARIS) || defined(RT_OS_DARWIN)
4928 /*
4929 * 'pszTrunk' contains just the interface name required in ring-0, while 'pszBridgedIfName' contains
4930 * interface name + optional description. We must not pass any description to the VM as it can differ
4931 * for the same interface name, eg: "nge0 - ethernet" (GUI) vs "nge0" (VBoxManage).
4932 */
4933 RTStrPrintf(szNetwork, sizeof(szNetwork), "HostInterfaceNetworking-%s", pszTrunk);
4934#else
4935 RTStrPrintf(szNetwork, sizeof(szNetwork), "HostInterfaceNetworking-%s", pszBridgedIfName);
4936#endif
4937 InsertConfigString(pCfg, "Network", szNetwork);
4938 networkName = Bstr(szNetwork);
4939 trunkName = Bstr(pszTrunk);
4940 trunkType = Bstr(TRUNKTYPE_NETFLT);
4941
4942# if defined(RT_OS_DARWIN)
4943 /** @todo Come up with a better deal here. Problem is that IHostNetworkInterface is completely useless here. */
4944 if ( strstr(pszBridgedIfName, "Wireless")
4945 || strstr(pszBridgedIfName, "AirPort" ))
4946 InsertConfigInteger(pCfg, "SharedMacOnWire", true);
4947# elif defined(RT_OS_LINUX)
4948 int iSock = socket(AF_INET, SOCK_DGRAM, 0);
4949 if (iSock >= 0)
4950 {
4951 struct iwreq WRq;
4952
4953 RT_ZERO(WRq);
4954 strncpy(WRq.ifr_name, pszBridgedIfName, IFNAMSIZ);
4955 bool fSharedMacOnWire = ioctl(iSock, SIOCGIWNAME, &WRq) >= 0;
4956 close(iSock);
4957 if (fSharedMacOnWire)
4958 {
4959 InsertConfigInteger(pCfg, "SharedMacOnWire", true);
4960 Log(("Set SharedMacOnWire\n"));
4961 }
4962 else
4963 Log(("Failed to get wireless name\n"));
4964 }
4965 else
4966 Log(("Failed to open wireless socket\n"));
4967# elif defined(RT_OS_FREEBSD)
4968 int iSock = socket(AF_INET, SOCK_DGRAM, 0);
4969 if (iSock >= 0)
4970 {
4971 struct ieee80211req WReq;
4972 uint8_t abData[32];
4973
4974 RT_ZERO(WReq);
4975 strncpy(WReq.i_name, pszBridgedIfName, sizeof(WReq.i_name));
4976 WReq.i_type = IEEE80211_IOC_SSID;
4977 WReq.i_val = -1;
4978 WReq.i_data = abData;
4979 WReq.i_len = sizeof(abData);
4980
4981 bool fSharedMacOnWire = ioctl(iSock, SIOCG80211, &WReq) >= 0;
4982 close(iSock);
4983 if (fSharedMacOnWire)
4984 {
4985 InsertConfigInteger(pCfg, "SharedMacOnWire", true);
4986 Log(("Set SharedMacOnWire\n"));
4987 }
4988 else
4989 Log(("Failed to get wireless name\n"));
4990 }
4991 else
4992 Log(("Failed to open wireless socket\n"));
4993# elif defined(RT_OS_WINDOWS)
4994# define DEVNAME_PREFIX L"\\\\.\\"
4995 /* we are getting the medium type via IOCTL_NDIS_QUERY_GLOBAL_STATS Io Control
4996 * there is a pretty long way till there though since we need to obtain the symbolic link name
4997 * for the adapter device we are going to query given the device Guid */
4998
4999
5000 /* prepend the "\\\\.\\" to the bind name to obtain the link name */
5001
5002 wchar_t FileName[MAX_PATH];
5003 wcscpy(FileName, DEVNAME_PREFIX);
5004 wcscpy((wchar_t*)(((char*)FileName) + sizeof(DEVNAME_PREFIX) - sizeof(FileName[0])), pswzBindName);
5005
5006 /* open the device */
5007 HANDLE hDevice = CreateFile(FileName,
5008 GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE,
5009 NULL,
5010 OPEN_EXISTING,
5011 FILE_ATTRIBUTE_NORMAL,
5012 NULL);
5013
5014 if (hDevice != INVALID_HANDLE_VALUE)
5015 {
5016 bool fSharedMacOnWire = false;
5017
5018 /* now issue the OID_GEN_PHYSICAL_MEDIUM query */
5019 DWORD Oid = OID_GEN_PHYSICAL_MEDIUM;
5020 NDIS_PHYSICAL_MEDIUM PhMedium;
5021 DWORD cbResult;
5022 if (DeviceIoControl(hDevice,
5023 IOCTL_NDIS_QUERY_GLOBAL_STATS,
5024 &Oid,
5025 sizeof(Oid),
5026 &PhMedium,
5027 sizeof(PhMedium),
5028 &cbResult,
5029 NULL))
5030 {
5031 /* that was simple, now examine PhMedium */
5032 if ( PhMedium == NdisPhysicalMediumWirelessWan
5033 || PhMedium == NdisPhysicalMediumWirelessLan
5034 || PhMedium == NdisPhysicalMediumNative802_11
5035 || PhMedium == NdisPhysicalMediumBluetooth)
5036 fSharedMacOnWire = true;
5037 }
5038 else
5039 {
5040 int winEr = GetLastError();
5041 LogRel(("Console::configNetwork: DeviceIoControl failed, err (0x%x), ignoring\n", winEr));
5042 Assert(winEr == ERROR_INVALID_PARAMETER || winEr == ERROR_NOT_SUPPORTED || winEr == ERROR_BAD_COMMAND);
5043 }
5044 CloseHandle(hDevice);
5045
5046 if (fSharedMacOnWire)
5047 {
5048 Log(("this is a wireless adapter"));
5049 InsertConfigInteger(pCfg, "SharedMacOnWire", true);
5050 Log(("Set SharedMacOnWire\n"));
5051 }
5052 else
5053 Log(("this is NOT a wireless adapter"));
5054 }
5055 else
5056 {
5057 int winEr = GetLastError();
5058 AssertLogRelMsgFailed(("Console::configNetwork: CreateFile failed, err (0x%x), ignoring\n", winEr));
5059 }
5060
5061 CoTaskMemFree(pswzBindName);
5062
5063 pAdaptorComponent.setNull();
5064 /* release the pNc finally */
5065 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
5066# else
5067 /** @todo PORTME: wireless detection */
5068# endif
5069
5070# if defined(RT_OS_SOLARIS)
5071# if 0 /* bird: this is a bit questionable and might cause more trouble than its worth. */
5072 /* Zone access restriction, don't allow snooping the global zone. */
5073 zoneid_t ZoneId = getzoneid();
5074 if (ZoneId != GLOBAL_ZONEID)
5075 {
5076 InsertConfigInteger(pCfg, "IgnoreAllPromisc", true);
5077 }
5078# endif
5079# endif
5080
5081#elif defined(RT_OS_WINDOWS) /* not defined NetFlt */
5082 /* NOTHING TO DO HERE */
5083#elif defined(RT_OS_LINUX)
5084/// @todo aleksey: is there anything to be done here?
5085#elif defined(RT_OS_FREEBSD)
5086/** @todo FreeBSD: Check out this later (HIF networking). */
5087#else
5088# error "Port me"
5089#endif
5090 break;
5091 }
5092
5093 case NetworkAttachmentType_Internal:
5094 {
5095 hrc = aNetworkAdapter->COMGETTER(InternalNetwork)(bstr.asOutParam()); H();
5096 if (!bstr.isEmpty())
5097 {
5098 InsertConfigString(pLunL0, "Driver", "IntNet");
5099 InsertConfigNode(pLunL0, "Config", &pCfg);
5100 InsertConfigString(pCfg, "Network", bstr);
5101 InsertConfigInteger(pCfg, "TrunkType", kIntNetTrunkType_WhateverNone);
5102 InsertConfigString(pCfg, "IfPolicyPromisc", pszPromiscuousGuestPolicy);
5103 networkName = bstr;
5104 trunkType = Bstr(TRUNKTYPE_WHATEVER);
5105 }
5106 break;
5107 }
5108
5109 case NetworkAttachmentType_HostOnly:
5110 {
5111 InsertConfigString(pLunL0, "Driver", "IntNet");
5112 InsertConfigNode(pLunL0, "Config", &pCfg);
5113
5114 Bstr HostOnlyName;
5115 hrc = aNetworkAdapter->COMGETTER(HostOnlyInterface)(HostOnlyName.asOutParam());
5116 if (FAILED(hrc))
5117 {
5118 LogRel(("NetworkAttachmentType_HostOnly: COMGETTER(HostOnlyInterface) failed, hrc (0x%x)\n", hrc));
5119 H();
5120 }
5121
5122 Utf8Str HostOnlyNameUtf8(HostOnlyName);
5123 const char *pszHostOnlyName = HostOnlyNameUtf8.c_str();
5124 ComPtr<IHostNetworkInterface> hostInterface;
5125 rc = host->FindHostNetworkInterfaceByName(HostOnlyName.raw(),
5126 hostInterface.asOutParam());
5127 if (!SUCCEEDED(rc))
5128 {
5129 LogRel(("NetworkAttachmentType_HostOnly: FindByName failed, rc (0x%x)\n", rc));
5130 return VMSetError(VMR3GetVM(mpUVM), VERR_INTERNAL_ERROR, RT_SRC_POS,
5131 N_("Nonexistent host networking interface, name '%ls'"),
5132 HostOnlyName.raw());
5133 }
5134
5135 char szNetwork[INTNET_MAX_NETWORK_NAME];
5136 RTStrPrintf(szNetwork, sizeof(szNetwork), "HostInterfaceNetworking-%s", pszHostOnlyName);
5137
5138#if defined(RT_OS_WINDOWS)
5139# ifndef VBOX_WITH_NETFLT
5140 hrc = E_NOTIMPL;
5141 LogRel(("NetworkAttachmentType_HostOnly: Not Implemented\n"));
5142 H();
5143# else /* defined VBOX_WITH_NETFLT*/
5144 /** @todo r=bird: Put this in a function. */
5145
5146 HostNetworkInterfaceType_T eIfType;
5147 hrc = hostInterface->COMGETTER(InterfaceType)(&eIfType);
5148 if (FAILED(hrc))
5149 {
5150 LogRel(("NetworkAttachmentType_HostOnly: COMGETTER(InterfaceType) failed, hrc (0x%x)\n", hrc));
5151 H();
5152 }
5153
5154 if (eIfType != HostNetworkInterfaceType_HostOnly)
5155 return VMSetError(VMR3GetVM(mpUVM), VERR_INTERNAL_ERROR, RT_SRC_POS,
5156 N_("Interface ('%ls') is not a Host-Only Adapter interface"),
5157 HostOnlyName.raw());
5158
5159 hrc = hostInterface->COMGETTER(Id)(bstr.asOutParam());
5160 if (FAILED(hrc))
5161 {
5162 LogRel(("NetworkAttachmentType_HostOnly: COMGETTER(Id) failed, hrc (0x%x)\n", hrc));
5163 H();
5164 }
5165 Guid hostIFGuid(bstr);
5166
5167 INetCfg *pNc;
5168 ComPtr<INetCfgComponent> pAdaptorComponent;
5169 LPWSTR pszApp;
5170 hrc = VBoxNetCfgWinQueryINetCfg(&pNc, FALSE, L"VirtualBox", 10, &pszApp);
5171 Assert(hrc == S_OK);
5172 if (hrc != S_OK)
5173 {
5174 LogRel(("NetworkAttachmentType_HostOnly: Failed to get NetCfg, hrc=%Rhrc (0x%x)\n", hrc, hrc));
5175 H();
5176 }
5177
5178 /* get the adapter's INetCfgComponent*/
5179 hrc = VBoxNetCfgWinGetComponentByGuid(pNc, &GUID_DEVCLASS_NET, (GUID*)hostIFGuid.raw(),
5180 pAdaptorComponent.asOutParam());
5181 if (hrc != S_OK)
5182 {
5183 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
5184 LogRel(("NetworkAttachmentType_HostOnly: VBoxNetCfgWinGetComponentByGuid failed, hrc=%Rhrc (0x%x)\n", hrc, hrc));
5185 H();
5186 }
5187# define VBOX_WIN_BINDNAME_PREFIX "\\DEVICE\\"
5188 char szTrunkName[INTNET_MAX_TRUNK_NAME];
5189 char *pszTrunkName = szTrunkName;
5190 wchar_t * pswzBindName;
5191 hrc = pAdaptorComponent->GetBindName(&pswzBindName);
5192 Assert(hrc == S_OK);
5193 if (hrc == S_OK)
5194 {
5195 int cwBindName = (int)wcslen(pswzBindName) + 1;
5196 int cbFullBindNamePrefix = sizeof(VBOX_WIN_BINDNAME_PREFIX);
5197 if (sizeof(szTrunkName) > cbFullBindNamePrefix + cwBindName)
5198 {
5199 strcpy(szTrunkName, VBOX_WIN_BINDNAME_PREFIX);
5200 pszTrunkName += cbFullBindNamePrefix-1;
5201 if (!WideCharToMultiByte(CP_ACP, 0, pswzBindName, cwBindName, pszTrunkName,
5202 sizeof(szTrunkName) - cbFullBindNamePrefix + 1, NULL, NULL))
5203 {
5204 DWORD err = GetLastError();
5205 hrc = HRESULT_FROM_WIN32(err);
5206 AssertLogRelMsgFailed(("NetworkAttachmentType_HostOnly: WideCharToMultiByte failed, hr=%Rhrc (0x%x) err=%u\n",
5207 hrc, hrc, err));
5208 }
5209 }
5210 else
5211 {
5212 AssertLogRelMsgFailed(("NetworkAttachmentType_HostOnly: insufficient szTrunkName buffer space\n"));
5213 /** @todo set appropriate error code */
5214 hrc = E_FAIL;
5215 }
5216
5217 if (hrc != S_OK)
5218 {
5219 AssertFailed();
5220 CoTaskMemFree(pswzBindName);
5221 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
5222 H();
5223 }
5224 }
5225 else
5226 {
5227 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
5228 AssertLogRelMsgFailed(("NetworkAttachmentType_HostOnly: VBoxNetCfgWinGetComponentByGuid failed, hrc=%Rhrc (0x%x)\n",
5229 hrc, hrc));
5230 H();
5231 }
5232
5233
5234 CoTaskMemFree(pswzBindName);
5235
5236 /* Assume we should use the old NDIS5.1 version of driver which uses TRUNKTYPE_NETADP */
5237 trunkType = TRUNKTYPE_NETADP;
5238
5239 HKEY hkParams;
5240 hrc = pAdaptorComponent->OpenParamKey(&hkParams);
5241 Assert(hrc == S_OK);
5242 if (hrc == S_OK)
5243 {
5244 WCHAR swzInfSection[16];
5245 DWORD dwSize = sizeof(swzInfSection);
5246 hrc = RegQueryValueExW(hkParams, L"InfSection", NULL, NULL, (LPBYTE)swzInfSection, &dwSize);
5247 if (hrc == S_OK)
5248 {
5249 if (!_wcsnicmp(swzInfSection, L"VBoxNetAdp6.ndi", sizeof(L"VBoxNetAdp6.ndi")/2))
5250 {
5251 /*
5252 * This is NDIS 6.x miniport, it relies on NetLwf filter to
5253 * run actual traffic. We use netflt attachment instead of
5254 * netadp, which is used in case of NDIS 5.x.
5255 */
5256 trunkType = TRUNKTYPE_NETFLT;
5257 }
5258 }
5259 RegCloseKey(hkParams);
5260 }
5261 else
5262 {
5263 LogRel(("Console::i_configNetwork: INetCfgComponent::GetId(%s) failed, err (0x%x), "
5264 "falling back to NDIS5 attachment\n", pszTrunkName, hrc));
5265 /* Nothing to do here as the trunk type defaults to NETADP */
5266 }
5267 InsertConfigInteger(pCfg, "TrunkType", trunkType == TRUNKTYPE_NETFLT ? kIntNetTrunkType_NetFlt : kIntNetTrunkType_NetAdp);
5268
5269 pAdaptorComponent.setNull();
5270 /* release the pNc finally */
5271 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
5272
5273 const char *pszTrunk = szTrunkName;
5274
5275 InsertConfigString(pCfg, "Trunk", pszTrunk);
5276 InsertConfigString(pCfg, "Network", szNetwork);
5277 InsertConfigInteger(pCfg, "IgnoreConnectFailure", (uint64_t)fIgnoreConnectFailure); /** @todo why is this
5278 windows only?? */
5279 networkName = Bstr(szNetwork);
5280 trunkName = Bstr(pszTrunk);
5281# endif /* defined VBOX_WITH_NETFLT*/
5282#elif defined(RT_OS_DARWIN)
5283 InsertConfigString(pCfg, "Trunk", pszHostOnlyName);
5284 InsertConfigString(pCfg, "Network", szNetwork);
5285 InsertConfigInteger(pCfg, "TrunkType", kIntNetTrunkType_NetAdp);
5286 networkName = Bstr(szNetwork);
5287 trunkName = Bstr(pszHostOnlyName);
5288 trunkType = TRUNKTYPE_NETADP;
5289#else
5290 InsertConfigString(pCfg, "Trunk", pszHostOnlyName);
5291 InsertConfigString(pCfg, "Network", szNetwork);
5292 InsertConfigInteger(pCfg, "TrunkType", kIntNetTrunkType_NetFlt);
5293 networkName = Bstr(szNetwork);
5294 trunkName = Bstr(pszHostOnlyName);
5295 trunkType = TRUNKTYPE_NETFLT;
5296#endif
5297 InsertConfigString(pCfg, "IfPolicyPromisc", pszPromiscuousGuestPolicy);
5298
5299#if !defined(RT_OS_WINDOWS) && defined(VBOX_WITH_NETFLT)
5300
5301 Bstr tmpAddr, tmpMask;
5302
5303 hrc = virtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPAddress",
5304 pszHostOnlyName).raw(),
5305 tmpAddr.asOutParam());
5306 if (SUCCEEDED(hrc) && !tmpAddr.isEmpty())
5307 {
5308 hrc = virtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPNetMask",
5309 pszHostOnlyName).raw(),
5310 tmpMask.asOutParam());
5311 if (SUCCEEDED(hrc) && !tmpMask.isEmpty())
5312 hrc = hostInterface->EnableStaticIPConfig(tmpAddr.raw(),
5313 tmpMask.raw());
5314 else
5315 hrc = hostInterface->EnableStaticIPConfig(tmpAddr.raw(),
5316 Bstr(VBOXNET_IPV4MASK_DEFAULT).raw());
5317 }
5318 else
5319 {
5320 /* Grab the IP number from the 'vboxnetX' instance number (see netif.h) */
5321 hrc = hostInterface->EnableStaticIPConfig(getDefaultIPv4Address(Bstr(pszHostOnlyName)).raw(),
5322 Bstr(VBOXNET_IPV4MASK_DEFAULT).raw());
5323 }
5324
5325 ComAssertComRC(hrc); /** @todo r=bird: Why this isn't fatal? (H()) */
5326
5327 hrc = virtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPV6Address",
5328 pszHostOnlyName).raw(),
5329 tmpAddr.asOutParam());
5330 if (SUCCEEDED(hrc))
5331 hrc = virtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPV6NetMask", pszHostOnlyName).raw(),
5332 tmpMask.asOutParam());
5333 if (SUCCEEDED(hrc) && !tmpAddr.isEmpty() && !tmpMask.isEmpty())
5334 {
5335 hrc = hostInterface->EnableStaticIPConfigV6(tmpAddr.raw(),
5336 Utf8Str(tmpMask).toUInt32());
5337 ComAssertComRC(hrc); /** @todo r=bird: Why this isn't fatal? (H()) */
5338 }
5339#endif
5340 break;
5341 }
5342
5343 case NetworkAttachmentType_Generic:
5344 {
5345 hrc = aNetworkAdapter->COMGETTER(GenericDriver)(bstr.asOutParam()); H();
5346 SafeArray<BSTR> names;
5347 SafeArray<BSTR> values;
5348 hrc = aNetworkAdapter->GetProperties(Bstr().raw(),
5349 ComSafeArrayAsOutParam(names),
5350 ComSafeArrayAsOutParam(values)); H();
5351
5352 InsertConfigString(pLunL0, "Driver", bstr);
5353 InsertConfigNode(pLunL0, "Config", &pCfg);
5354 for (size_t ii = 0; ii < names.size(); ++ii)
5355 {
5356 if (values[ii] && *values[ii])
5357 {
5358 Utf8Str name = names[ii];
5359 Utf8Str value = values[ii];
5360 InsertConfigString(pCfg, name.c_str(), value);
5361 }
5362 }
5363 break;
5364 }
5365
5366 case NetworkAttachmentType_NATNetwork:
5367 {
5368 hrc = aNetworkAdapter->COMGETTER(NATNetwork)(bstr.asOutParam()); H();
5369 if (!bstr.isEmpty())
5370 {
5371 /** @todo add intnet prefix to separate namespaces, and add trunk if dealing with vboxnatX */
5372 InsertConfigString(pLunL0, "Driver", "IntNet");
5373 InsertConfigNode(pLunL0, "Config", &pCfg);
5374 InsertConfigString(pCfg, "Network", bstr);
5375 InsertConfigInteger(pCfg, "TrunkType", kIntNetTrunkType_WhateverNone);
5376 InsertConfigString(pCfg, "IfPolicyPromisc", pszPromiscuousGuestPolicy);
5377 networkName = bstr;
5378 trunkType = Bstr(TRUNKTYPE_WHATEVER);
5379 }
5380 break;
5381 }
5382
5383 default:
5384 AssertMsgFailed(("should not get here!\n"));
5385 break;
5386 }
5387
5388 /*
5389 * Attempt to attach the driver.
5390 */
5391 switch (eAttachmentType)
5392 {
5393 case NetworkAttachmentType_Null:
5394 break;
5395
5396 case NetworkAttachmentType_Bridged:
5397 case NetworkAttachmentType_Internal:
5398 case NetworkAttachmentType_HostOnly:
5399 case NetworkAttachmentType_NAT:
5400 case NetworkAttachmentType_Generic:
5401 case NetworkAttachmentType_NATNetwork:
5402 {
5403 if (SUCCEEDED(hrc) && SUCCEEDED(rc))
5404 {
5405 if (fAttachDetach)
5406 {
5407 rc = PDMR3DriverAttach(mpUVM, pszDevice, uInstance, uLun, 0 /*fFlags*/, NULL /* ppBase */);
5408 //AssertRC(rc);
5409 }
5410
5411 {
5412 /** @todo pritesh: get the dhcp server name from the
5413 * previous network configuration and then stop the server
5414 * else it may conflict with the dhcp server running with
5415 * the current attachment type
5416 */
5417 /* Stop the hostonly DHCP Server */
5418 }
5419
5420 /*
5421 * NAT networks start their DHCP server theirself, see NATNetwork::Start()
5422 */
5423 if ( !networkName.isEmpty()
5424 && eAttachmentType != NetworkAttachmentType_NATNetwork)
5425 {
5426 /*
5427 * Until we implement service reference counters DHCP Server will be stopped
5428 * by DHCPServerRunner destructor.
5429 */
5430 ComPtr<IDHCPServer> dhcpServer;
5431 hrc = virtualBox->FindDHCPServerByNetworkName(networkName.raw(),
5432 dhcpServer.asOutParam());
5433 if (SUCCEEDED(hrc))
5434 {
5435 /* there is a DHCP server available for this network */
5436 BOOL fEnabledDhcp;
5437 hrc = dhcpServer->COMGETTER(Enabled)(&fEnabledDhcp);
5438 if (FAILED(hrc))
5439 {
5440 LogRel(("DHCP svr: COMGETTER(Enabled) failed, hrc (%Rhrc)\n", hrc));
5441 H();
5442 }
5443
5444 if (fEnabledDhcp)
5445 hrc = dhcpServer->Start(networkName.raw(),
5446 trunkName.raw(),
5447 trunkType.raw());
5448 }
5449 else
5450 hrc = S_OK;
5451 }
5452 }
5453
5454 break;
5455 }
5456
5457 default:
5458 AssertMsgFailed(("should not get here!\n"));
5459 break;
5460 }
5461
5462 meAttachmentType[uInstance] = eAttachmentType;
5463 }
5464 catch (ConfigError &x)
5465 {
5466 // InsertConfig threw something:
5467 return x.m_vrc;
5468 }
5469
5470#undef H
5471
5472 return VINF_SUCCESS;
5473}
5474
5475#ifdef VBOX_WITH_GUEST_PROPS
5476/**
5477 * Set an array of guest properties
5478 */
5479static void configSetProperties(VMMDev * const pVMMDev,
5480 void *names,
5481 void *values,
5482 void *timestamps,
5483 void *flags)
5484{
5485 VBOXHGCMSVCPARM parms[4];
5486
5487 parms[0].type = VBOX_HGCM_SVC_PARM_PTR;
5488 parms[0].u.pointer.addr = names;
5489 parms[0].u.pointer.size = 0; /* We don't actually care. */
5490 parms[1].type = VBOX_HGCM_SVC_PARM_PTR;
5491 parms[1].u.pointer.addr = values;
5492 parms[1].u.pointer.size = 0; /* We don't actually care. */
5493 parms[2].type = VBOX_HGCM_SVC_PARM_PTR;
5494 parms[2].u.pointer.addr = timestamps;
5495 parms[2].u.pointer.size = 0; /* We don't actually care. */
5496 parms[3].type = VBOX_HGCM_SVC_PARM_PTR;
5497 parms[3].u.pointer.addr = flags;
5498 parms[3].u.pointer.size = 0; /* We don't actually care. */
5499
5500 pVMMDev->hgcmHostCall("VBoxGuestPropSvc",
5501 guestProp::SET_PROPS_HOST,
5502 4,
5503 &parms[0]);
5504}
5505
5506/**
5507 * Set a single guest property
5508 */
5509static void configSetProperty(VMMDev * const pVMMDev,
5510 const char *pszName,
5511 const char *pszValue,
5512 const char *pszFlags)
5513{
5514 VBOXHGCMSVCPARM parms[4];
5515
5516 AssertPtrReturnVoid(pszName);
5517 AssertPtrReturnVoid(pszValue);
5518 AssertPtrReturnVoid(pszFlags);
5519 parms[0].type = VBOX_HGCM_SVC_PARM_PTR;
5520 parms[0].u.pointer.addr = (void *)pszName;
5521 parms[0].u.pointer.size = (uint32_t)strlen(pszName) + 1;
5522 parms[1].type = VBOX_HGCM_SVC_PARM_PTR;
5523 parms[1].u.pointer.addr = (void *)pszValue;
5524 parms[1].u.pointer.size = (uint32_t)strlen(pszValue) + 1;
5525 parms[2].type = VBOX_HGCM_SVC_PARM_PTR;
5526 parms[2].u.pointer.addr = (void *)pszFlags;
5527 parms[2].u.pointer.size = (uint32_t)strlen(pszFlags) + 1;
5528 pVMMDev->hgcmHostCall("VBoxGuestPropSvc", guestProp::SET_PROP_HOST, 3,
5529 &parms[0]);
5530}
5531
5532/**
5533 * Set the global flags value by calling the service
5534 * @returns the status returned by the call to the service
5535 *
5536 * @param pTable the service instance handle
5537 * @param eFlags the flags to set
5538 */
5539int configSetGlobalPropertyFlags(VMMDev * const pVMMDev,
5540 guestProp::ePropFlags eFlags)
5541{
5542 VBOXHGCMSVCPARM paParm;
5543 paParm.setUInt32(eFlags);
5544 int rc = pVMMDev->hgcmHostCall("VBoxGuestPropSvc",
5545 guestProp::SET_GLOBAL_FLAGS_HOST, 1,
5546 &paParm);
5547 if (RT_FAILURE(rc))
5548 {
5549 char szFlags[guestProp::MAX_FLAGS_LEN];
5550 if (RT_FAILURE(writeFlags(eFlags, szFlags)))
5551 Log(("Failed to set the global flags.\n"));
5552 else
5553 Log(("Failed to set the global flags \"%s\".\n", szFlags));
5554 }
5555 return rc;
5556}
5557#endif /* VBOX_WITH_GUEST_PROPS */
5558
5559/**
5560 * Set up the Guest Property service, populate it with properties read from
5561 * the machine XML and set a couple of initial properties.
5562 */
5563/* static */ int Console::i_configGuestProperties(void *pvConsole, PUVM pUVM)
5564{
5565#ifdef VBOX_WITH_GUEST_PROPS
5566 AssertReturn(pvConsole, VERR_INVALID_POINTER);
5567 ComObjPtr<Console> pConsole = static_cast<Console *>(pvConsole);
5568 AssertReturn(pConsole->m_pVMMDev, VERR_INVALID_POINTER);
5569
5570 /* Load the service */
5571 int rc = pConsole->m_pVMMDev->hgcmLoadService("VBoxGuestPropSvc", "VBoxGuestPropSvc");
5572
5573 if (RT_FAILURE(rc))
5574 {
5575 LogRel(("VBoxGuestPropSvc is not available. rc = %Rrc\n", rc));
5576 /* That is not a fatal failure. */
5577 rc = VINF_SUCCESS;
5578 }
5579 else
5580 {
5581 /*
5582 * Initialize built-in properties that can be changed and saved.
5583 *
5584 * These are typically transient properties that the guest cannot
5585 * change.
5586 */
5587
5588 {
5589 VBOXHGCMSVCPARM Params[2];
5590 int rc2 = pConsole->m_pVMMDev->hgcmHostCall("VBoxGuestPropSvc", guestProp::GET_DBGF_INFO_FN, 2, &Params[0]);
5591 if (RT_SUCCESS(rc2))
5592 {
5593 PFNDBGFHANDLEREXT pfnHandler = (PFNDBGFHANDLEREXT)(uintptr_t)Params[0].u.pointer.addr;
5594 void *pService = (void*)Params[1].u.pointer.addr;
5595 DBGFR3InfoRegisterExternal(pUVM, "guestprops", "Display the guest properties", pfnHandler, pService);
5596 }
5597 }
5598
5599 /* Sysprep execution by VBoxService. */
5600 configSetProperty(pConsole->m_pVMMDev,
5601 "/VirtualBox/HostGuest/SysprepExec", "",
5602 "TRANSIENT, RDONLYGUEST");
5603 configSetProperty(pConsole->m_pVMMDev,
5604 "/VirtualBox/HostGuest/SysprepArgs", "",
5605 "TRANSIENT, RDONLYGUEST");
5606
5607 /*
5608 * Pull over the properties from the server.
5609 */
5610 SafeArray<BSTR> namesOut;
5611 SafeArray<BSTR> valuesOut;
5612 SafeArray<LONG64> timestampsOut;
5613 SafeArray<BSTR> flagsOut;
5614 HRESULT hrc;
5615 hrc = pConsole->mControl->PullGuestProperties(ComSafeArrayAsOutParam(namesOut),
5616 ComSafeArrayAsOutParam(valuesOut),
5617 ComSafeArrayAsOutParam(timestampsOut),
5618 ComSafeArrayAsOutParam(flagsOut));
5619 AssertLogRelMsgReturn(SUCCEEDED(hrc), ("hrc=%Rhrc\n", hrc), VERR_MAIN_CONFIG_CONSTRUCTOR_COM_ERROR);
5620 size_t cProps = namesOut.size();
5621 size_t cAlloc = cProps + 1;
5622 if ( valuesOut.size() != cProps
5623 || timestampsOut.size() != cProps
5624 || flagsOut.size() != cProps
5625 )
5626 AssertFailedReturn(VERR_INVALID_PARAMETER);
5627
5628 char **papszNames, **papszValues, **papszFlags;
5629 char szEmpty[] = "";
5630 LONG64 *pai64Timestamps;
5631 papszNames = (char **)RTMemTmpAllocZ(sizeof(void *) * cAlloc);
5632 papszValues = (char **)RTMemTmpAllocZ(sizeof(void *) * cAlloc);
5633 pai64Timestamps = (LONG64 *)RTMemTmpAllocZ(sizeof(LONG64) * cAlloc);
5634 papszFlags = (char **)RTMemTmpAllocZ(sizeof(void *) * cAlloc);
5635 if (papszNames && papszValues && pai64Timestamps && papszFlags)
5636 {
5637 for (unsigned i = 0; RT_SUCCESS(rc) && i < cProps; ++i)
5638 {
5639 AssertPtrReturn(namesOut[i], VERR_INVALID_PARAMETER);
5640 rc = RTUtf16ToUtf8(namesOut[i], &papszNames[i]);
5641 if (RT_FAILURE(rc))
5642 break;
5643 if (valuesOut[i])
5644 rc = RTUtf16ToUtf8(valuesOut[i], &papszValues[i]);
5645 else
5646 papszValues[i] = szEmpty;
5647 if (RT_FAILURE(rc))
5648 break;
5649 pai64Timestamps[i] = timestampsOut[i];
5650 if (flagsOut[i])
5651 rc = RTUtf16ToUtf8(flagsOut[i], &papszFlags[i]);
5652 else
5653 papszFlags[i] = szEmpty;
5654 }
5655 if (RT_SUCCESS(rc))
5656 configSetProperties(pConsole->m_pVMMDev,
5657 (void *)papszNames,
5658 (void *)papszValues,
5659 (void *)pai64Timestamps,
5660 (void *)papszFlags);
5661 for (unsigned i = 0; i < cProps; ++i)
5662 {
5663 RTStrFree(papszNames[i]);
5664 if (valuesOut[i])
5665 RTStrFree(papszValues[i]);
5666 if (flagsOut[i])
5667 RTStrFree(papszFlags[i]);
5668 }
5669 }
5670 else
5671 rc = VERR_NO_MEMORY;
5672 RTMemTmpFree(papszNames);
5673 RTMemTmpFree(papszValues);
5674 RTMemTmpFree(pai64Timestamps);
5675 RTMemTmpFree(papszFlags);
5676 AssertRCReturn(rc, rc);
5677
5678 /*
5679 * These properties have to be set before pulling over the properties
5680 * from the machine XML, to ensure that properties saved in the XML
5681 * will override them.
5682 */
5683 /* Set the raw VBox version string as a guest property. Used for host/guest
5684 * version comparison. */
5685 configSetProperty(pConsole->m_pVMMDev, "/VirtualBox/HostInfo/VBoxVer",
5686 VBOX_VERSION_STRING_RAW, "TRANSIENT, RDONLYGUEST");
5687 /* Set the full VBox version string as a guest property. Can contain vendor-specific
5688 * information/branding and/or pre-release tags. */
5689 configSetProperty(pConsole->m_pVMMDev, "/VirtualBox/HostInfo/VBoxVerExt",
5690 VBOX_VERSION_STRING, "TRANSIENT, RDONLYGUEST");
5691 /* Set the VBox SVN revision as a guest property */
5692 configSetProperty(pConsole->m_pVMMDev, "/VirtualBox/HostInfo/VBoxRev",
5693 RTBldCfgRevisionStr(), "TRANSIENT, RDONLYGUEST");
5694
5695 /*
5696 * Register the host notification callback
5697 */
5698 HGCMSVCEXTHANDLE hDummy;
5699 HGCMHostRegisterServiceExtension(&hDummy, "VBoxGuestPropSvc",
5700 Console::i_doGuestPropNotification,
5701 pvConsole);
5702
5703#ifdef VBOX_WITH_GUEST_PROPS_RDONLY_GUEST
5704 rc = configSetGlobalPropertyFlags(pConsole->m_pVMMDev,
5705 guestProp::RDONLYGUEST);
5706 AssertRCReturn(rc, rc);
5707#endif
5708
5709 Log(("Set VBoxGuestPropSvc property store\n"));
5710 }
5711 return VINF_SUCCESS;
5712#else /* !VBOX_WITH_GUEST_PROPS */
5713 return VERR_NOT_SUPPORTED;
5714#endif /* !VBOX_WITH_GUEST_PROPS */
5715}
5716
5717/**
5718 * Set up the Guest Control service.
5719 */
5720/* static */ int Console::i_configGuestControl(void *pvConsole)
5721{
5722#ifdef VBOX_WITH_GUEST_CONTROL
5723 AssertReturn(pvConsole, VERR_INVALID_POINTER);
5724 ComObjPtr<Console> pConsole = static_cast<Console *>(pvConsole);
5725
5726 /* Load the service */
5727 int rc = pConsole->m_pVMMDev->hgcmLoadService("VBoxGuestControlSvc", "VBoxGuestControlSvc");
5728
5729 if (RT_FAILURE(rc))
5730 {
5731 LogRel(("VBoxGuestControlSvc is not available. rc = %Rrc\n", rc));
5732 /* That is not a fatal failure. */
5733 rc = VINF_SUCCESS;
5734 }
5735 else
5736 {
5737 HGCMSVCEXTHANDLE hDummy;
5738 rc = HGCMHostRegisterServiceExtension(&hDummy, "VBoxGuestControlSvc",
5739 &Guest::i_notifyCtrlDispatcher,
5740 pConsole->i_getGuest());
5741 if (RT_FAILURE(rc))
5742 Log(("Cannot register VBoxGuestControlSvc extension!\n"));
5743 else
5744 LogRel(("Guest Control service loaded\n"));
5745 }
5746
5747 return rc;
5748#else /* !VBOX_WITH_GUEST_CONTROL */
5749 return VERR_NOT_SUPPORTED;
5750#endif /* !VBOX_WITH_GUEST_CONTROL */
5751}
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