VirtualBox

source: vbox/trunk/src/VBox/Main/ConsoleImpl2.cpp@ 23454

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

Main/ConsoleImpl2: fix creating CFGM tree creation for DVD/floppy drives with no medium

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 127.9 KB
Line 
1/* $Id: ConsoleImpl2.cpp 23418 2009-09-29 16:18:20Z vboxsync $ */
2/** @file
3 * VBox Console COM Class implementation
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-2009 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
23 * Clara, CA 95054 USA or visit http://www.sun.com if you need
24 * additional information or have any questions.
25 */
26
27/*******************************************************************************
28* Header Files *
29*******************************************************************************/
30#include "ConsoleImpl.h"
31#include "DisplayImpl.h"
32#include "VMMDev.h"
33
34// generated header
35#include "SchemaDefs.h"
36
37#include "Logging.h"
38
39#include <iprt/buildconfig.h>
40#include <iprt/string.h>
41#include <iprt/path.h>
42#include <iprt/dir.h>
43#include <iprt/param.h>
44#if 0 /* enable to play with lots of memory. */
45# include <iprt/env.h>
46#endif
47
48#include <VBox/vmapi.h>
49#include <VBox/err.h>
50#include <VBox/version.h>
51#include <VBox/HostServices/VBoxClipboardSvc.h>
52#ifdef VBOX_WITH_CROGL
53#include <VBox/HostServices/VBoxCrOpenGLSvc.h>
54#endif
55#ifdef VBOX_WITH_GUEST_PROPS
56# include <VBox/HostServices/GuestPropertySvc.h>
57# include <VBox/com/defs.h>
58# include <VBox/com/array.h>
59# include <hgcm/HGCM.h> /** @todo it should be possible to register a service
60 * extension using a VMMDev callback. */
61# include <vector>
62#endif /* VBOX_WITH_GUEST_PROPS */
63#include <VBox/intnet.h>
64
65#include <VBox/com/string.h>
66#include <VBox/com/array.h>
67
68#if defined(RT_OS_SOLARIS) && defined(VBOX_WITH_NETFLT)
69# include <zone.h>
70#endif
71
72#if defined(RT_OS_LINUX) && defined(VBOX_WITH_NETFLT)
73# include <unistd.h>
74# include <sys/ioctl.h>
75# include <sys/socket.h>
76# include <linux/types.h>
77# include <linux/if.h>
78# include <linux/wireless.h>
79#endif
80
81#if defined(RT_OS_FREEBSD) && defined(VBOX_WITH_NETFLT)
82# include <unistd.h>
83# include <sys/types.h>
84# include <sys/ioctl.h>
85# include <sys/socket.h>
86# include <net/if.h>
87#endif
88
89#if defined(RT_OS_WINDOWS) && defined(VBOX_WITH_NETFLT)
90# include <VBox/WinNetConfig.h>
91# include <Ntddndis.h>
92# include <devguid.h>
93#endif
94
95#if !defined(RT_OS_WINDOWS) && defined(VBOX_WITH_NETFLT)
96# include <HostNetworkInterfaceImpl.h>
97# include <netif.h>
98#endif
99
100#include "DHCPServerRunner.h"
101
102#include <VBox/param.h>
103
104/* Comment out the following line to remove VMWare compatibility hack. */
105#define VMWARE_NET_IN_SLOT_11
106
107/**
108 * Translate IDE StorageControllerType_T to string representation.
109 */
110const char* controllerString(StorageControllerType_T enmType)
111{
112 switch (enmType)
113 {
114 case StorageControllerType_PIIX3:
115 return "PIIX3";
116 case StorageControllerType_PIIX4:
117 return "PIIX4";
118 case StorageControllerType_ICH6:
119 return "ICH6";
120 default:
121 return "Unknown";
122 }
123}
124
125/*
126 * VC++ 8 / amd64 has some serious trouble with this function.
127 * As a temporary measure, we'll drop global optimizations.
128 */
129#if defined(_MSC_VER) && defined(RT_ARCH_AMD64)
130# pragma optimize("g", off)
131#endif
132
133/**
134 * Construct the VM configuration tree (CFGM).
135 *
136 * This is a callback for VMR3Create() call. It is called from CFGMR3Init()
137 * in the emulation thread (EMT). Any per thread COM/XPCOM initialization
138 * is done here.
139 *
140 * @param pVM VM handle.
141 * @param pvConsole Pointer to the VMPowerUpTask object.
142 * @return VBox status code.
143 *
144 * @note Locks the Console object for writing.
145 */
146DECLCALLBACK(int) Console::configConstructor(PVM pVM, void *pvConsole)
147{
148 LogFlowFuncEnter();
149 /* Note: hardcoded assumption about number of slots; see rom bios */
150 bool afPciDeviceNo[32] = {false};
151 bool fFdcEnabled = false;
152
153#if !defined (VBOX_WITH_XPCOM)
154 {
155 /* initialize COM */
156 HRESULT hrc = CoInitializeEx(NULL,
157 COINIT_MULTITHREADED | COINIT_DISABLE_OLE1DDE |
158 COINIT_SPEED_OVER_MEMORY);
159 LogFlow (("Console::configConstructor(): CoInitializeEx()=%08X\n", hrc));
160 AssertComRCReturn (hrc, VERR_GENERAL_FAILURE);
161 }
162#endif
163
164 AssertReturn(pvConsole, VERR_GENERAL_FAILURE);
165 ComObjPtr<Console> pConsole = static_cast <Console *> (pvConsole);
166
167 AutoCaller autoCaller(pConsole);
168 AssertComRCReturn (autoCaller.rc(), VERR_ACCESS_DENIED);
169
170 /* lock the console because we widely use internal fields and methods */
171 AutoWriteLock alock(pConsole);
172
173 /* Save the VM pointer in the machine object */
174 pConsole->mpVM = pVM;
175
176 ComPtr<IMachine> pMachine = pConsole->machine();
177
178 int rc;
179 HRESULT hrc;
180 BSTR str = NULL;
181
182#define STR_FREE() do { if (str) { SysFreeString(str); str = NULL; } } while (0)
183#define RC_CHECK() do { if (RT_FAILURE(rc)) { AssertMsgFailed(("rc=%Rrc\n", rc)); STR_FREE(); return rc; } } while (0)
184#define H() do { if (FAILED(hrc)) { AssertMsgFailed(("hrc=%#x\n", hrc)); STR_FREE(); return VERR_GENERAL_FAILURE; } } while (0)
185
186 /*
187 * Get necessary objects and frequently used parameters.
188 */
189 ComPtr<IVirtualBox> virtualBox;
190 hrc = pMachine->COMGETTER(Parent)(virtualBox.asOutParam()); H();
191
192 ComPtr<IHost> host;
193 hrc = virtualBox->COMGETTER(Host)(host.asOutParam()); H();
194
195 ComPtr<ISystemProperties> systemProperties;
196 hrc = virtualBox->COMGETTER(SystemProperties)(systemProperties.asOutParam()); H();
197
198 ComPtr<IBIOSSettings> biosSettings;
199 hrc = pMachine->COMGETTER(BIOSSettings)(biosSettings.asOutParam()); H();
200
201 hrc = pMachine->COMGETTER(Id)(&str); H();
202 Guid MachineUuid(str);
203 PCRTUUID pUuid = MachineUuid.raw();
204 STR_FREE();
205
206 ULONG cRamMBs;
207 hrc = pMachine->COMGETTER(MemorySize)(&cRamMBs); H();
208#if 0 /* enable to play with lots of memory. */
209 if (RTEnvExist("VBOX_RAM_SIZE"))
210 cRamMBs = RTStrToUInt64(RTEnvGet("VBOX_RAM_SIZE"));
211#endif
212 uint64_t const cbRam = cRamMBs * (uint64_t)_1M;
213 uint32_t const cbRamHole = MM_RAM_HOLE_SIZE_DEFAULT;
214
215 ULONG cCpus = 1;
216 hrc = pMachine->COMGETTER(CPUCount)(&cCpus); H();
217
218 Bstr osTypeId;
219 hrc = pMachine->COMGETTER(OSTypeId)(osTypeId.asOutParam()); H();
220
221 BOOL fIOAPIC;
222 hrc = biosSettings->COMGETTER(IOAPICEnabled)(&fIOAPIC); H();
223
224 /*
225 * Get root node first.
226 * This is the only node in the tree.
227 */
228 PCFGMNODE pRoot = CFGMR3GetRoot(pVM);
229 Assert(pRoot);
230
231 /*
232 * Set the root (and VMM) level values.
233 */
234 hrc = pMachine->COMGETTER(Name)(&str); H();
235 rc = CFGMR3InsertStringW(pRoot, "Name", str); RC_CHECK();
236 rc = CFGMR3InsertBytes(pRoot, "UUID", pUuid, sizeof(*pUuid)); RC_CHECK();
237 rc = CFGMR3InsertInteger(pRoot, "RamSize", cbRam); RC_CHECK();
238 rc = CFGMR3InsertInteger(pRoot, "RamHoleSize", cbRamHole); RC_CHECK();
239 rc = CFGMR3InsertInteger(pRoot, "NumCPUs", cCpus); RC_CHECK();
240 rc = CFGMR3InsertInteger(pRoot, "TimerMillies", 10); RC_CHECK();
241 rc = CFGMR3InsertInteger(pRoot, "RawR3Enabled", 1); /* boolean */ RC_CHECK();
242 rc = CFGMR3InsertInteger(pRoot, "RawR0Enabled", 1); /* boolean */ RC_CHECK();
243 /** @todo Config: RawR0, PATMEnabled and CASMEnabled needs attention later. */
244 rc = CFGMR3InsertInteger(pRoot, "PATMEnabled", 1); /* boolean */ RC_CHECK();
245 rc = CFGMR3InsertInteger(pRoot, "CSAMEnabled", 1); /* boolean */ RC_CHECK();
246
247 if (osTypeId == "WindowsNT4")
248 {
249 /*
250 * We must limit CPUID count for Windows NT 4, as otherwise it stops
251 * with error 0x3e (MULTIPROCESSOR_CONFIGURATION_NOT_SUPPORTED).
252 */
253 LogRel(("Limiting CPUID leaf count for NT4 guests\n"));
254 PCFGMNODE pCPUM;
255 rc = CFGMR3InsertNode(pRoot, "CPUM", &pCPUM); RC_CHECK();
256 rc = CFGMR3InsertInteger(pCPUM, "NT4LeafLimit", true); RC_CHECK();
257 }
258
259 /* hardware virtualization extensions */
260 BOOL fHWVirtExEnabled;
261 hrc = pMachine->COMGETTER(HWVirtExEnabled)(&fHWVirtExEnabled); H();
262 if (cCpus > 1) /** @todo SMP: This isn't nice, but things won't work on mac otherwise. */
263 fHWVirtExEnabled = TRUE;
264
265#ifdef RT_OS_DARWIN
266 rc = CFGMR3InsertInteger(pRoot, "HwVirtExtForced", fHWVirtExEnabled); RC_CHECK();
267#else
268 /* - With more than 4GB PGM will use different RAMRANGE sizes for raw
269 mode and hv mode to optimize lookup times.
270 - With more than one virtual CPU, raw-mode isn't a fallback option. */
271 BOOL fHwVirtExtForced = fHWVirtExEnabled
272 && ( cbRam > (_4G - cbRamHole)
273 || cCpus > 1);
274 rc = CFGMR3InsertInteger(pRoot, "HwVirtExtForced", fHwVirtExtForced); RC_CHECK();
275#endif
276
277 PCFGMNODE pHWVirtExt;
278 rc = CFGMR3InsertNode(pRoot, "HWVirtExt", &pHWVirtExt); RC_CHECK();
279 if (fHWVirtExEnabled)
280 {
281 rc = CFGMR3InsertInteger(pHWVirtExt, "Enabled", 1); RC_CHECK();
282
283 /* Indicate whether 64-bit guests are supported or not. */
284 /** @todo This is currently only forced off on 32-bit hosts only because it
285 * makes a lof of difference there (REM and Solaris performance).
286 */
287
288 ComPtr<IGuestOSType> guestOSType;
289 hrc = virtualBox->GetGuestOSType(osTypeId, guestOSType.asOutParam()); H();
290
291 BOOL fSupportsLongMode = false;
292 hrc = host->GetProcessorFeature(ProcessorFeature_LongMode,
293 &fSupportsLongMode); H();
294 BOOL fIs64BitGuest = false;
295 hrc = guestOSType->COMGETTER(Is64Bit)(&fIs64BitGuest); H();
296
297 if (fSupportsLongMode && fIs64BitGuest)
298 {
299 rc = CFGMR3InsertInteger(pHWVirtExt, "64bitEnabled", 1); RC_CHECK();
300#if ARCH_BITS == 32 /* The recompiler must use VBoxREM64 (32-bit host only). */
301 PCFGMNODE pREM;
302 rc = CFGMR3InsertNode(pRoot, "REM", &pREM); RC_CHECK();
303 rc = CFGMR3InsertInteger(pREM, "64bitEnabled", 1); RC_CHECK();
304#endif
305 }
306#if ARCH_BITS == 32 /* 32-bit guests only. */
307 else
308 {
309 rc = CFGMR3InsertInteger(pHWVirtExt, "64bitEnabled", 0); RC_CHECK();
310 }
311#endif
312
313 /* @todo Not exactly pretty to check strings; VBOXOSTYPE would be better, but that requires quite a bit of API change in Main. */
314 if ( !fIs64BitGuest
315 && fIOAPIC
316 && ( osTypeId == "WindowsNT4"
317 || osTypeId == "Windows2000"
318 || osTypeId == "WindowsXP"
319 || osTypeId == "Windows2003"))
320 {
321 /* Only allow TPR patching for NT, Win2k, XP and Windows Server 2003. (32 bits mode)
322 * We may want to consider adding more guest OSes (Solaris) later on.
323 */
324 rc = CFGMR3InsertInteger(pHWVirtExt, "TPRPatchingEnabled", 1); RC_CHECK();
325 }
326 }
327
328 /* Nested paging (VT-x/AMD-V) */
329 BOOL fEnableNestedPaging = false;
330 hrc = pMachine->COMGETTER(HWVirtExNestedPagingEnabled)(&fEnableNestedPaging); H();
331 rc = CFGMR3InsertInteger(pRoot, "EnableNestedPaging", fEnableNestedPaging); RC_CHECK();
332
333 /* VPID (VT-x) */
334 BOOL fEnableVPID = false;
335 hrc = pMachine->COMGETTER(HWVirtExVPIDEnabled)(&fEnableVPID); H();
336 rc = CFGMR3InsertInteger(pRoot, "EnableVPID", fEnableVPID); RC_CHECK();
337
338 /* Physical Address Extension (PAE) */
339 BOOL fEnablePAE = false;
340 hrc = pMachine->COMGETTER(PAEEnabled)(&fEnablePAE); H();
341 rc = CFGMR3InsertInteger(pRoot, "EnablePAE", fEnablePAE); RC_CHECK();
342
343 BOOL fPXEDebug;
344 hrc = biosSettings->COMGETTER(PXEDebugEnabled)(&fPXEDebug); H();
345
346 /*
347 * PDM config.
348 * Load drivers in VBoxC.[so|dll]
349 */
350 PCFGMNODE pPDM;
351 PCFGMNODE pDrivers;
352 PCFGMNODE pMod;
353 rc = CFGMR3InsertNode(pRoot, "PDM", &pPDM); RC_CHECK();
354 rc = CFGMR3InsertNode(pPDM, "Drivers", &pDrivers); RC_CHECK();
355 rc = CFGMR3InsertNode(pDrivers, "VBoxC", &pMod); RC_CHECK();
356#ifdef VBOX_WITH_XPCOM
357 // VBoxC is located in the components subdirectory
358 char szPathVBoxC[RTPATH_MAX];
359 rc = RTPathAppPrivateArch(szPathVBoxC, RTPATH_MAX - sizeof("/components/VBoxC")); AssertRC(rc);
360 strcat(szPathVBoxC, "/components/VBoxC");
361 rc = CFGMR3InsertString(pMod, "Path", szPathVBoxC); RC_CHECK();
362#else
363 rc = CFGMR3InsertString(pMod, "Path", "VBoxC"); RC_CHECK();
364#endif
365
366 /*
367 * Devices
368 */
369 PCFGMNODE pDevices = NULL; /* /Devices */
370 PCFGMNODE pDev = NULL; /* /Devices/Dev/ */
371 PCFGMNODE pInst = NULL; /* /Devices/Dev/0/ */
372 PCFGMNODE pCfg = NULL; /* /Devices/Dev/.../Config/ */
373 PCFGMNODE pLunL0 = NULL; /* /Devices/Dev/0/LUN#0/ */
374 PCFGMNODE pLunL1 = NULL; /* /Devices/Dev/0/LUN#0/AttachedDriver/ */
375 PCFGMNODE pLunL2 = NULL; /* /Devices/Dev/0/LUN#0/AttachedDriver/Config/ */
376 PCFGMNODE pBiosCfg = NULL; /* /Devices/pcbios/0/Config/ */
377
378 rc = CFGMR3InsertNode(pRoot, "Devices", &pDevices); RC_CHECK();
379
380 /*
381 * PC Arch.
382 */
383 rc = CFGMR3InsertNode(pDevices, "pcarch", &pDev); RC_CHECK();
384 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
385 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
386 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
387
388 /*
389 * The time offset
390 */
391 LONG64 timeOffset;
392 hrc = biosSettings->COMGETTER(TimeOffset)(&timeOffset); H();
393 PCFGMNODE pTMNode;
394 rc = CFGMR3InsertNode(pRoot, "TM", &pTMNode); RC_CHECK();
395 rc = CFGMR3InsertInteger(pTMNode, "UTCOffset", timeOffset * 1000000); RC_CHECK();
396
397 /*
398 * DMA
399 */
400 rc = CFGMR3InsertNode(pDevices, "8237A", &pDev); RC_CHECK();
401 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
402 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
403
404 /*
405 * PCI buses.
406 */
407 rc = CFGMR3InsertNode(pDevices, "pci", &pDev); /* piix3 */ RC_CHECK();
408 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
409 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
410 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
411 rc = CFGMR3InsertInteger(pCfg, "IOAPIC", fIOAPIC); RC_CHECK();
412
413#if 0 /* enable this to test PCI bridging */
414 rc = CFGMR3InsertNode(pDevices, "pcibridge", &pDev); RC_CHECK();
415 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
416 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
417 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
418 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", 14); RC_CHECK();
419 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
420 rc = CFGMR3InsertInteger(pInst, "PCIBusNo", 0);/* -> pci[0] */ RC_CHECK();
421
422 rc = CFGMR3InsertNode(pDev, "1", &pInst); RC_CHECK();
423 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
424 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
425 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", 1); RC_CHECK();
426 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
427 rc = CFGMR3InsertInteger(pInst, "PCIBusNo", 1);/* ->pcibridge[0] */ RC_CHECK();
428
429 rc = CFGMR3InsertNode(pDev, "2", &pInst); RC_CHECK();
430 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
431 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
432 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", 3); RC_CHECK();
433 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
434 rc = CFGMR3InsertInteger(pInst, "PCIBusNo", 1);/* ->pcibridge[0] */ RC_CHECK();
435#endif
436
437 /*
438 * Temporary hack for enabling the next three devices and various ACPI features.
439 */
440 Bstr tmpStr2;
441 hrc = pMachine->GetExtraData(Bstr("VBoxInternal2/SupportExtHwProfile"), tmpStr2.asOutParam()); H();
442 BOOL fExtProfile = tmpStr2 == Bstr("on");
443
444 /*
445 * High Precision Event Timer (HPET)
446 */
447 BOOL fHpetEnabled;
448#ifdef VBOX_WITH_HPET
449 fHpetEnabled = fExtProfile;
450#else
451 fHpetEnabled = false;
452#endif
453 if (fHpetEnabled)
454 {
455 rc = CFGMR3InsertNode(pDevices, "hpet", &pDev); RC_CHECK();
456 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
457 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
458 }
459
460 /*
461 * System Management Controller (SMC)
462 */
463 BOOL fSmcEnabled;
464#ifdef VBOX_WITH_SMC
465 fSmcEnabled = fExtProfile;
466#else
467 fSmcEnabled = false;
468#endif
469 if (fSmcEnabled)
470 {
471 rc = CFGMR3InsertNode(pDevices, "smc", &pDev); RC_CHECK();
472 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
473 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
474 }
475
476 /*
477 * Low Pin Count (LPC) bus
478 */
479 BOOL fLpcEnabled;
480 /** @todo: implement appropriate getter */
481#ifdef VBOX_WITH_LPC
482 fLpcEnabled = fExtProfile;
483#else
484 fLpcEnabled = false;
485#endif
486 if (fLpcEnabled)
487 {
488 rc = CFGMR3InsertNode(pDevices, "lpc", &pDev); RC_CHECK();
489 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
490 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
491 }
492
493 /*
494 * PS/2 keyboard & mouse.
495 */
496 rc = CFGMR3InsertNode(pDevices, "pckbd", &pDev); RC_CHECK();
497 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
498 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
499 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
500
501 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
502 rc = CFGMR3InsertString(pLunL0, "Driver", "KeyboardQueue"); RC_CHECK();
503 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
504 rc = CFGMR3InsertInteger(pCfg, "QueueSize", 64); RC_CHECK();
505
506 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL1); RC_CHECK();
507 rc = CFGMR3InsertString(pLunL1, "Driver", "MainKeyboard"); RC_CHECK();
508 rc = CFGMR3InsertNode(pLunL1, "Config", &pCfg); RC_CHECK();
509 Keyboard *pKeyboard = pConsole->mKeyboard;
510 rc = CFGMR3InsertInteger(pCfg, "Object", (uintptr_t)pKeyboard); RC_CHECK();
511
512 rc = CFGMR3InsertNode(pInst, "LUN#1", &pLunL0); RC_CHECK();
513 rc = CFGMR3InsertString(pLunL0, "Driver", "MouseQueue"); RC_CHECK();
514 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
515 rc = CFGMR3InsertInteger(pCfg, "QueueSize", 128); RC_CHECK();
516
517 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL1); RC_CHECK();
518 rc = CFGMR3InsertString(pLunL1, "Driver", "MainMouse"); RC_CHECK();
519 rc = CFGMR3InsertNode(pLunL1, "Config", &pCfg); RC_CHECK();
520 Mouse *pMouse = pConsole->mMouse;
521 rc = CFGMR3InsertInteger(pCfg, "Object", (uintptr_t)pMouse); RC_CHECK();
522
523 /*
524 * i8254 Programmable Interval Timer And Dummy Speaker
525 */
526 rc = CFGMR3InsertNode(pDevices, "i8254", &pDev); RC_CHECK();
527 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
528 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
529#ifdef DEBUG
530 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
531#endif
532
533 /*
534 * i8259 Programmable Interrupt Controller.
535 */
536 rc = CFGMR3InsertNode(pDevices, "i8259", &pDev); RC_CHECK();
537 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
538 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
539 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
540
541 /*
542 * Advanced Programmable Interrupt Controller.
543 * SMP: Each CPU has a LAPIC, but we have a single device representing all LAPICs states,
544 * thus only single insert
545 */
546 rc = CFGMR3InsertNode(pDevices, "apic", &pDev); RC_CHECK();
547 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
548 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
549 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
550 rc = CFGMR3InsertInteger(pCfg, "IOAPIC", fIOAPIC); RC_CHECK();
551 rc = CFGMR3InsertInteger(pCfg, "NumCPUs", cCpus); RC_CHECK();
552
553 if (fIOAPIC)
554 {
555 /*
556 * I/O Advanced Programmable Interrupt Controller.
557 */
558 rc = CFGMR3InsertNode(pDevices, "ioapic", &pDev); RC_CHECK();
559 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
560 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
561 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
562 }
563
564 /*
565 * RTC MC146818.
566 */
567 rc = CFGMR3InsertNode(pDevices, "mc146818", &pDev); RC_CHECK();
568 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
569 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
570
571 /*
572 * VGA.
573 */
574 rc = CFGMR3InsertNode(pDevices, "vga", &pDev); RC_CHECK();
575 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
576 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
577 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", 2); RC_CHECK();
578 Assert(!afPciDeviceNo[2]);
579 afPciDeviceNo[2] = true;
580 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
581 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
582 ULONG cVRamMBs;
583 hrc = pMachine->COMGETTER(VRAMSize)(&cVRamMBs); H();
584 rc = CFGMR3InsertInteger(pCfg, "VRamSize", cVRamMBs * _1M); RC_CHECK();
585 ULONG cMonitorCount;
586 hrc = pMachine->COMGETTER(MonitorCount)(&cMonitorCount); H();
587 rc = CFGMR3InsertInteger(pCfg, "MonitorCount", cMonitorCount); RC_CHECK();
588#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE /* not safe here yet. */ /** @todo this needs fixing !!! No wonder VGA is slooooooooow on 32-bit darwin! */
589 rc = CFGMR3InsertInteger(pCfg, "R0Enabled", fHWVirtExEnabled); RC_CHECK();
590#endif
591
592 /*
593 * BIOS logo
594 */
595 BOOL fFadeIn;
596 hrc = biosSettings->COMGETTER(LogoFadeIn)(&fFadeIn); H();
597 rc = CFGMR3InsertInteger(pCfg, "FadeIn", fFadeIn ? 1 : 0); RC_CHECK();
598 BOOL fFadeOut;
599 hrc = biosSettings->COMGETTER(LogoFadeOut)(&fFadeOut); H();
600 rc = CFGMR3InsertInteger(pCfg, "FadeOut", fFadeOut ? 1: 0); RC_CHECK();
601 ULONG logoDisplayTime;
602 hrc = biosSettings->COMGETTER(LogoDisplayTime)(&logoDisplayTime); H();
603 rc = CFGMR3InsertInteger(pCfg, "LogoTime", logoDisplayTime); RC_CHECK();
604 Bstr logoImagePath;
605 hrc = biosSettings->COMGETTER(LogoImagePath)(logoImagePath.asOutParam()); H();
606 rc = CFGMR3InsertString(pCfg, "LogoFile", logoImagePath ? Utf8Str(logoImagePath).c_str() : ""); RC_CHECK();
607
608 /*
609 * Boot menu
610 */
611 BIOSBootMenuMode_T eBootMenuMode;
612 int iShowBootMenu;
613 biosSettings->COMGETTER(BootMenuMode)(&eBootMenuMode);
614 switch (eBootMenuMode)
615 {
616 case BIOSBootMenuMode_Disabled: iShowBootMenu = 0; break;
617 case BIOSBootMenuMode_MenuOnly: iShowBootMenu = 1; break;
618 default: iShowBootMenu = 2; break;
619 }
620 rc = CFGMR3InsertInteger(pCfg, "ShowBootMenu", iShowBootMenu); RC_CHECK();
621
622 /* Custom VESA mode list */
623 unsigned cModes = 0;
624 for (unsigned iMode = 1; iMode <= 16; ++iMode)
625 {
626 char szExtraDataKey[sizeof("CustomVideoModeXX")];
627 RTStrPrintf(szExtraDataKey, sizeof(szExtraDataKey), "CustomVideoMode%u", iMode);
628 hrc = pMachine->GetExtraData(Bstr(szExtraDataKey), &str); H();
629 if (!str || !*str)
630 break;
631 rc = CFGMR3InsertStringW(pCfg, szExtraDataKey, str); RC_CHECK();
632 STR_FREE();
633 ++cModes;
634 }
635 STR_FREE();
636 rc = CFGMR3InsertInteger(pCfg, "CustomVideoModes", cModes);
637
638 /* VESA height reduction */
639 ULONG ulHeightReduction;
640 IFramebuffer *pFramebuffer = pConsole->getDisplay()->getFramebuffer();
641 if (pFramebuffer)
642 {
643 hrc = pFramebuffer->COMGETTER(HeightReduction)(&ulHeightReduction); H();
644 }
645 else
646 {
647 /* If framebuffer is not available, there is no height reduction. */
648 ulHeightReduction = 0;
649 }
650 rc = CFGMR3InsertInteger(pCfg, "HeightReduction", ulHeightReduction); RC_CHECK();
651
652 /* Attach the display. */
653 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
654 rc = CFGMR3InsertString(pLunL0, "Driver", "MainDisplay"); RC_CHECK();
655 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
656 Display *pDisplay = pConsole->mDisplay;
657 rc = CFGMR3InsertInteger(pCfg, "Object", (uintptr_t)pDisplay); RC_CHECK();
658
659 /*
660 * Storage controllers.
661 */
662 com::SafeIfaceArray<IStorageController> ctrls;
663 hrc = pMachine->COMGETTER(StorageControllers)(ComSafeArrayAsOutParam(ctrls)); H();
664
665 for (size_t i = 0; i < ctrls.size(); ++ i)
666 {
667 PCFGMNODE pCtlInst = NULL; /* /Devices/<name>/0/ */
668 StorageControllerType_T enmCtrlType;
669 StorageBus_T enmBus;
670 bool fSCSI = false;
671 BSTR controllerName;
672
673 rc = ctrls[i]->COMGETTER(ControllerType)(&enmCtrlType); H();
674 rc = ctrls[i]->COMGETTER(Bus)(&enmBus); H();
675 rc = ctrls[i]->COMGETTER(Name)(&controllerName); H();
676
677 const char *pszCtrlDev = pConsole->controllerTypeToDev(enmCtrlType);
678
679 rc = CFGMR3InsertNode(pDevices, pszCtrlDev, &pDev); RC_CHECK();
680 /** @todo support multiple instances of a controller */
681 rc = CFGMR3InsertNode(pDev, "0", &pCtlInst); RC_CHECK();
682
683 switch (enmCtrlType)
684 {
685 case StorageControllerType_LsiLogic:
686 {
687 rc = CFGMR3InsertInteger(pCtlInst, "Trusted", 1); RC_CHECK();
688 rc = CFGMR3InsertInteger(pCtlInst, "PCIDeviceNo", 20); RC_CHECK();
689 Assert(!afPciDeviceNo[20]);
690 afPciDeviceNo[20] = true;
691 rc = CFGMR3InsertInteger(pCtlInst, "PCIFunctionNo", 0); RC_CHECK();
692 rc = CFGMR3InsertNode(pCtlInst, "Config", &pCfg); RC_CHECK();
693 fSCSI = true;
694
695 /* Attach the status driver */
696 rc = CFGMR3InsertNode(pCtlInst, "LUN#999", &pLunL0); RC_CHECK();
697 rc = CFGMR3InsertString(pLunL0, "Driver", "MainStatus"); RC_CHECK();
698 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
699 rc = CFGMR3InsertInteger(pCfg, "papLeds", (uintptr_t)&pConsole->mapSCSILeds[0]); RC_CHECK();
700 rc = CFGMR3InsertInteger(pCfg, "First", 0); RC_CHECK();
701 rc = CFGMR3InsertInteger(pCfg, "Last", 15); RC_CHECK();
702 break;
703 }
704
705 case StorageControllerType_BusLogic:
706 {
707 rc = CFGMR3InsertInteger(pCtlInst, "Trusted", 1); RC_CHECK();
708 rc = CFGMR3InsertInteger(pCtlInst, "PCIDeviceNo", 21); RC_CHECK();
709 Assert(!afPciDeviceNo[21]);
710 afPciDeviceNo[21] = true;
711 rc = CFGMR3InsertInteger(pCtlInst, "PCIFunctionNo", 0); RC_CHECK();
712 rc = CFGMR3InsertNode(pCtlInst, "Config", &pCfg); RC_CHECK();
713 fSCSI = true;
714
715 /* Attach the status driver */
716 rc = CFGMR3InsertNode(pCtlInst, "LUN#999", &pLunL0); RC_CHECK();
717 rc = CFGMR3InsertString(pLunL0, "Driver", "MainStatus"); RC_CHECK();
718 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
719 rc = CFGMR3InsertInteger(pCfg, "papLeds", (uintptr_t)&pConsole->mapSCSILeds[0]); RC_CHECK();
720 rc = CFGMR3InsertInteger(pCfg, "First", 0); RC_CHECK();
721 rc = CFGMR3InsertInteger(pCfg, "Last", 15); RC_CHECK();
722 break;
723 }
724
725 case StorageControllerType_IntelAhci:
726 {
727 rc = CFGMR3InsertInteger(pCtlInst, "Trusted", 1); RC_CHECK();
728 rc = CFGMR3InsertInteger(pCtlInst, "PCIDeviceNo", 13); RC_CHECK();
729 Assert(!afPciDeviceNo[13]);
730 afPciDeviceNo[13] = true;
731 rc = CFGMR3InsertInteger(pCtlInst, "PCIFunctionNo", 0); RC_CHECK();
732 rc = CFGMR3InsertNode(pCtlInst, "Config", &pCfg); RC_CHECK();
733
734 ULONG cPorts = 0;
735 hrc = ctrls[i]->COMGETTER(PortCount)(&cPorts); H();
736 rc = CFGMR3InsertInteger(pCfg, "PortCount", cPorts); RC_CHECK();
737
738 /* Needed configuration values for the bios. */
739 if (pBiosCfg)
740 {
741 rc = CFGMR3InsertString(pBiosCfg, "SataHardDiskDevice", "ahci"); RC_CHECK();
742 }
743
744 for (uint32_t j = 0; j < 4; ++j)
745 {
746 static const char *s_apszConfig[4] =
747 { "PrimaryMaster", "PrimarySlave", "SecondaryMaster", "SecondarySlave" };
748 static const char *s_apszBiosConfig[4] =
749 { "SataPrimaryMasterLUN", "SataPrimarySlaveLUN", "SataSecondaryMasterLUN", "SataSecondarySlaveLUN" };
750
751 LONG lPortNumber = -1;
752 hrc = ctrls[i]->GetIDEEmulationPort(j, &lPortNumber); H();
753 rc = CFGMR3InsertInteger(pCfg, s_apszConfig[j], lPortNumber); RC_CHECK();
754 if (pBiosCfg)
755 {
756 rc = CFGMR3InsertInteger(pBiosCfg, s_apszBiosConfig[j], lPortNumber); RC_CHECK();
757 }
758 }
759
760 /* Attach the status driver */
761 rc = CFGMR3InsertNode(pCtlInst, "LUN#999", &pLunL0); RC_CHECK();
762 rc = CFGMR3InsertString(pLunL0, "Driver", "MainStatus"); RC_CHECK();
763 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
764 AssertRelease(cPorts <= RT_ELEMENTS(pConsole->mapSATALeds));
765 rc = CFGMR3InsertInteger(pCfg, "papLeds", (uintptr_t)&pConsole->mapSATALeds[0]); RC_CHECK();
766 rc = CFGMR3InsertInteger(pCfg, "First", 0); RC_CHECK();
767 rc = CFGMR3InsertInteger(pCfg, "Last", cPorts - 1); RC_CHECK();
768 break;
769 }
770
771 case StorageControllerType_PIIX3:
772 case StorageControllerType_PIIX4:
773 case StorageControllerType_ICH6:
774 {
775 /*
776 * IDE (update this when the main interface changes)
777 */
778 rc = CFGMR3InsertInteger(pCtlInst, "Trusted", 1); /* boolean */ RC_CHECK();
779 rc = CFGMR3InsertInteger(pCtlInst, "PCIDeviceNo", 1); RC_CHECK();
780 Assert(!afPciDeviceNo[1]);
781 afPciDeviceNo[1] = true;
782 rc = CFGMR3InsertInteger(pCtlInst, "PCIFunctionNo", 1); RC_CHECK();
783 rc = CFGMR3InsertNode(pCtlInst, "Config", &pCfg); RC_CHECK();
784 rc = CFGMR3InsertString(pCfg, "Type", controllerString(enmCtrlType)); RC_CHECK();
785
786 /* Attach the status driver */
787 rc = CFGMR3InsertNode(pCtlInst, "LUN#999", &pLunL0); RC_CHECK();
788 rc = CFGMR3InsertString(pLunL0, "Driver", "MainStatus"); RC_CHECK();
789 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
790 rc = CFGMR3InsertInteger(pCfg, "papLeds", (uintptr_t)&pConsole->mapIDELeds[0]);RC_CHECK();
791 rc = CFGMR3InsertInteger(pCfg, "First", 0); RC_CHECK();
792 rc = CFGMR3InsertInteger(pCfg, "Last", 3); RC_CHECK();
793 break;
794 }
795 case StorageControllerType_I82078:
796 {
797 /*
798 * i82078 Floppy drive controller
799 */
800 fFdcEnabled = true;
801 rc = CFGMR3InsertInteger(pCtlInst, "Trusted", 1); RC_CHECK();
802 rc = CFGMR3InsertNode(pCtlInst, "Config", &pCfg); RC_CHECK();
803 rc = CFGMR3InsertInteger(pCfg, "IRQ", 6); RC_CHECK();
804 rc = CFGMR3InsertInteger(pCfg, "DMA", 2); RC_CHECK();
805 rc = CFGMR3InsertInteger(pCfg, "MemMapped", 0 ); RC_CHECK();
806 rc = CFGMR3InsertInteger(pCfg, "IOBase", 0x3f0); RC_CHECK();
807
808 /* Attach the status driver */
809 rc = CFGMR3InsertNode(pCtlInst, "LUN#999", &pLunL0); RC_CHECK();
810 rc = CFGMR3InsertString(pLunL0, "Driver", "MainStatus"); RC_CHECK();
811 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
812 rc = CFGMR3InsertInteger(pCfg, "papLeds", (uintptr_t)&pConsole->mapFDLeds[0]); RC_CHECK();
813 rc = CFGMR3InsertInteger(pCfg, "First", 0); RC_CHECK();
814 rc = CFGMR3InsertInteger(pCfg, "Last", 0); RC_CHECK();
815 break;
816 }
817
818 default:
819 AssertMsgFailedReturn(("invalid storage controller type: %d\n", enmCtrlType), VERR_GENERAL_FAILURE);
820 }
821
822 /* At the moment we only support one controller per type. So the instance id is always 0. */
823 rc = ctrls[i]->COMSETTER(Instance)(0); H();
824
825 /* Attach the hard disks. */
826 com::SafeIfaceArray<IMediumAttachment> atts;
827 hrc = pMachine->GetMediumAttachmentsOfController(controllerName,
828 ComSafeArrayAsOutParam(atts)); H();
829
830 for (size_t j = 0; j < atts.size(); ++ j)
831 {
832 BOOL fHostDrive = FALSE;
833
834 ComPtr<IMedium> medium;
835 hrc = atts [j]->COMGETTER(Medium)(medium.asOutParam()); H();
836 LONG lDev;
837 hrc = atts[j]->COMGETTER(Device)(&lDev); H();
838 LONG lPort;
839 hrc = atts[j]->COMGETTER(Port)(&lPort); H();
840 DeviceType_T lType;
841 hrc = atts[j]->COMGETTER(Type)(&lType); H();
842
843 unsigned uLUN = 0;
844 hrc = pConsole->convertBusPortDeviceToLun(enmBus, lPort, lDev, uLUN); H();
845 rc = CFGMR3InsertNodeF(pCtlInst, &pLunL0, "LUN#%u", uLUN); RC_CHECK();
846
847 /* SCSI has a another driver between device and block. */
848 if (fSCSI)
849 {
850 rc = CFGMR3InsertString(pLunL0, "Driver", "SCSI"); RC_CHECK();
851 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
852
853 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL0); RC_CHECK();
854 }
855
856 if (!medium.isNull())
857 {
858 hrc = medium->COMGETTER(HostDrive)(&fHostDrive); H();
859 }
860
861 if (fHostDrive)
862 {
863 Assert(!medium.isNull());
864 if (lType == DeviceType_DVD)
865 {
866 rc = CFGMR3InsertString(pLunL0, "Driver", "HostDVD"); RC_CHECK();
867 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
868
869 hrc = medium->COMGETTER(Location)(&str); H();
870 rc = CFGMR3InsertStringW(pCfg, "Path", str); RC_CHECK();
871 STR_FREE();
872
873 BOOL fPassthrough = false;
874 hrc = atts[j]->COMGETTER(Passthrough)(&fPassthrough); H();
875 rc = CFGMR3InsertInteger(pCfg, "Passthrough", !!fPassthrough); RC_CHECK();
876 }
877 else if (lType == DeviceType_Floppy)
878 {
879 rc = CFGMR3InsertString(pLunL0, "Driver", "HostFloppy"); RC_CHECK();
880 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
881
882 hrc = medium->COMGETTER(Location)(&str); H();
883 rc = CFGMR3InsertStringW(pCfg, "Path", str); RC_CHECK();
884 STR_FREE();
885 }
886 }
887 else
888 {
889 rc = CFGMR3InsertString(pLunL0, "Driver", "Block"); RC_CHECK();
890 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
891 switch (lType)
892 {
893 case DeviceType_DVD:
894 rc = CFGMR3InsertString(pCfg, "Type", "DVD"); RC_CHECK();
895 rc = CFGMR3InsertInteger(pCfg, "Mountable", 1); RC_CHECK();
896 break;
897 case DeviceType_Floppy:
898 rc = CFGMR3InsertString(pCfg, "Type", "Floppy 1.44"); RC_CHECK();
899 rc = CFGMR3InsertInteger(pCfg, "Mountable", 1); RC_CHECK();
900 break;
901 case DeviceType_HardDisk:
902 default:
903 rc = CFGMR3InsertString(pCfg, "Type", "HardDisk"); RC_CHECK();
904 rc = CFGMR3InsertInteger(pCfg, "Mountable", 0); RC_CHECK();
905 }
906
907 if (!medium.isNull())
908 {
909 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL1); RC_CHECK();
910 rc = CFGMR3InsertString(pLunL1, "Driver", "VD"); RC_CHECK();
911 rc = CFGMR3InsertNode(pLunL1, "Config", &pCfg); RC_CHECK();
912
913 hrc = medium->COMGETTER(Location)(&str); H();
914 rc = CFGMR3InsertStringW(pCfg, "Path", str); RC_CHECK();
915 STR_FREE();
916
917 hrc = medium->COMGETTER(Format)(&str); H();
918 rc = CFGMR3InsertStringW(pCfg, "Format", str); RC_CHECK();
919 STR_FREE();
920
921 /* DVDs are always readonly */
922 if (lType == DeviceType_DVD)
923 {
924 rc = CFGMR3InsertInteger(pCfg, "ReadOnly", 1); RC_CHECK();
925 }
926
927 /* Pass all custom parameters. */
928 bool fHostIP = true;
929 SafeArray<BSTR> names;
930 SafeArray<BSTR> values;
931 hrc = medium->GetProperties(NULL,
932 ComSafeArrayAsOutParam(names),
933 ComSafeArrayAsOutParam(values)); H();
934
935 if (names.size() != 0)
936 {
937 PCFGMNODE pVDC;
938 rc = CFGMR3InsertNode(pCfg, "VDConfig", &pVDC); RC_CHECK();
939 for (size_t ii = 0; ii < names.size(); ++ii)
940 {
941 if (values[ii] && *values[ii])
942 {
943 Utf8Str name = names[ii];
944 Utf8Str value = values[ii];
945 rc = CFGMR3InsertString(pVDC, name.c_str(), value.c_str());
946 if ( name.compare("HostIPStack") == 0
947 && value.compare("0") == 0)
948 fHostIP = false;
949 }
950 }
951 }
952
953 /* Create an inversed tree of parents. */
954 ComPtr<IMedium> parentMedium = medium;
955 for (PCFGMNODE pParent = pCfg;;)
956 {
957 hrc = parentMedium->COMGETTER(Parent)(medium.asOutParam()); H();
958 if (medium.isNull())
959 break;
960
961 PCFGMNODE pCur;
962 rc = CFGMR3InsertNode(pParent, "Parent", &pCur); RC_CHECK();
963 hrc = medium->COMGETTER(Location)(&str); H();
964 rc = CFGMR3InsertStringW(pCur, "Path", str); RC_CHECK();
965 STR_FREE();
966
967 hrc = medium->COMGETTER(Format)(&str); H();
968 rc = CFGMR3InsertStringW(pCur, "Format", str); RC_CHECK();
969 STR_FREE();
970
971 /* Pass all custom parameters. */
972 SafeArray<BSTR> names;
973 SafeArray<BSTR> values;
974 hrc = medium->GetProperties(NULL,
975 ComSafeArrayAsOutParam(names),
976 ComSafeArrayAsOutParam(values)); H();
977
978 if (names.size() != 0)
979 {
980 PCFGMNODE pVDC;
981 rc = CFGMR3InsertNode(pCur, "VDConfig", &pVDC); RC_CHECK();
982 for (size_t ii = 0; ii < names.size(); ++ii)
983 {
984 if (values[ii])
985 {
986 Utf8Str name = names[ii];
987 Utf8Str value = values[ii];
988 rc = CFGMR3InsertString(pVDC, name.c_str(), value.c_str());
989 if ( name.compare("HostIPStack") == 0
990 && value.compare("0") == 0)
991 fHostIP = false;
992 }
993 }
994 }
995
996 /* Custom code: put marker to not use host IP stack to driver
997 * configuration node. Simplifies life of DrvVD a bit. */
998 if (!fHostIP)
999 {
1000 rc = CFGMR3InsertInteger(pCfg, "HostIPStack", 0); RC_CHECK();
1001 }
1002
1003 /* next */
1004 pParent = pCur;
1005 parentMedium = medium;
1006 }
1007 }
1008 }
1009 }
1010 H();
1011 }
1012 H();
1013
1014 /*
1015 * Network adapters
1016 */
1017#ifdef VMWARE_NET_IN_SLOT_11
1018 bool fSwapSlots3and11 = false;
1019#endif
1020 PCFGMNODE pDevPCNet = NULL; /* PCNet-type devices */
1021 rc = CFGMR3InsertNode(pDevices, "pcnet", &pDevPCNet); RC_CHECK();
1022#ifdef VBOX_WITH_E1000
1023 PCFGMNODE pDevE1000 = NULL; /* E1000-type devices */
1024 rc = CFGMR3InsertNode(pDevices, "e1000", &pDevE1000); RC_CHECK();
1025#endif
1026#ifdef VBOX_WITH_VIRTIO
1027 PCFGMNODE pDevVirtioNet = NULL; /* Virtio network devices */
1028 rc = CFGMR3InsertNode(pDevices, "virtio-net", &pDevVirtioNet); RC_CHECK();
1029#endif /* VBOX_WITH_VIRTIO */
1030 for (ULONG ulInstance = 0; ulInstance < SchemaDefs::NetworkAdapterCount; ++ulInstance)
1031 {
1032 ComPtr<INetworkAdapter> networkAdapter;
1033 hrc = pMachine->GetNetworkAdapter(ulInstance, networkAdapter.asOutParam()); H();
1034 BOOL fEnabled = FALSE;
1035 hrc = networkAdapter->COMGETTER(Enabled)(&fEnabled); H();
1036 if (!fEnabled)
1037 continue;
1038
1039 /*
1040 * The virtual hardware type. Create appropriate device first.
1041 */
1042 const char *pszAdapterName = "pcnet";
1043 NetworkAdapterType_T adapterType;
1044 hrc = networkAdapter->COMGETTER(AdapterType)(&adapterType); H();
1045 switch (adapterType)
1046 {
1047 case NetworkAdapterType_Am79C970A:
1048 case NetworkAdapterType_Am79C973:
1049 pDev = pDevPCNet;
1050 break;
1051#ifdef VBOX_WITH_E1000
1052 case NetworkAdapterType_I82540EM:
1053 case NetworkAdapterType_I82543GC:
1054 case NetworkAdapterType_I82545EM:
1055 pDev = pDevE1000;
1056 pszAdapterName = "e1000";
1057 break;
1058#endif
1059#ifdef VBOX_WITH_VIRTIO
1060 case NetworkAdapterType_Virtio:
1061 pDev = pDevVirtioNet;
1062 pszAdapterName = "virtio";
1063 break;
1064#endif /* VBOX_WITH_VIRTIO */
1065 default:
1066 AssertMsgFailed(("Invalid network adapter type '%d' for slot '%d'",
1067 adapterType, ulInstance));
1068 return VMSetError(pVM, VERR_INVALID_PARAMETER, RT_SRC_POS,
1069 N_("Invalid network adapter type '%d' for slot '%d'"),
1070 adapterType, ulInstance);
1071 }
1072
1073 rc = CFGMR3InsertNodeF(pDev, &pInst, "%u", ulInstance); RC_CHECK();
1074 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
1075 /* the first network card gets the PCI ID 3, the next 3 gets 8..10,
1076 * next 4 get 16..19. */
1077 unsigned iPciDeviceNo = 3;
1078 if (ulInstance)
1079 {
1080 if (ulInstance < 4)
1081 iPciDeviceNo = ulInstance - 1 + 8;
1082 else
1083 iPciDeviceNo = ulInstance - 4 + 16;
1084 }
1085#ifdef VMWARE_NET_IN_SLOT_11
1086 /*
1087 * Dirty hack for PCI slot compatibility with VMWare,
1088 * it assigns slot 11 to the first network controller.
1089 */
1090 if (iPciDeviceNo == 3 && adapterType == NetworkAdapterType_I82545EM)
1091 {
1092 iPciDeviceNo = 0x11;
1093 fSwapSlots3and11 = true;
1094 }
1095 else if (iPciDeviceNo == 0x11 && fSwapSlots3and11)
1096 iPciDeviceNo = 3;
1097#endif
1098 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", iPciDeviceNo); RC_CHECK();
1099 Assert(!afPciDeviceNo[iPciDeviceNo]);
1100 afPciDeviceNo[iPciDeviceNo] = true;
1101 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
1102 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1103#ifdef VBOX_WITH_2X_4GB_ADDR_SPACE /* not safe here yet. */
1104 if (pDev == pDevPCNet)
1105 {
1106 rc = CFGMR3InsertInteger(pCfg, "R0Enabled", false); RC_CHECK();
1107 }
1108#endif
1109
1110 /*
1111 * The virtual hardware type. PCNet supports two types.
1112 */
1113 switch (adapterType)
1114 {
1115 case NetworkAdapterType_Am79C970A:
1116 rc = CFGMR3InsertInteger(pCfg, "Am79C973", 0); RC_CHECK();
1117 break;
1118 case NetworkAdapterType_Am79C973:
1119 rc = CFGMR3InsertInteger(pCfg, "Am79C973", 1); RC_CHECK();
1120 break;
1121 case NetworkAdapterType_I82540EM:
1122 rc = CFGMR3InsertInteger(pCfg, "AdapterType", 0); RC_CHECK();
1123 break;
1124 case NetworkAdapterType_I82543GC:
1125 rc = CFGMR3InsertInteger(pCfg, "AdapterType", 1); RC_CHECK();
1126 break;
1127 case NetworkAdapterType_I82545EM:
1128 rc = CFGMR3InsertInteger(pCfg, "AdapterType", 2); RC_CHECK();
1129 break;
1130 }
1131
1132 /*
1133 * Get the MAC address and convert it to binary representation
1134 */
1135 Bstr macAddr;
1136 hrc = networkAdapter->COMGETTER(MACAddress)(macAddr.asOutParam()); H();
1137 Assert(macAddr);
1138 Utf8Str macAddrUtf8 = macAddr;
1139 char *macStr = (char*)macAddrUtf8.raw();
1140 Assert(strlen(macStr) == 12);
1141 RTMAC Mac;
1142 memset(&Mac, 0, sizeof(Mac));
1143 char *pMac = (char*)&Mac;
1144 for (uint32_t i = 0; i < 6; ++i)
1145 {
1146 char c1 = *macStr++ - '0';
1147 if (c1 > 9)
1148 c1 -= 7;
1149 char c2 = *macStr++ - '0';
1150 if (c2 > 9)
1151 c2 -= 7;
1152 *pMac++ = ((c1 & 0x0f) << 4) | (c2 & 0x0f);
1153 }
1154 rc = CFGMR3InsertBytes(pCfg, "MAC", &Mac, sizeof(Mac)); RC_CHECK();
1155
1156 /*
1157 * Check if the cable is supposed to be unplugged
1158 */
1159 BOOL fCableConnected;
1160 hrc = networkAdapter->COMGETTER(CableConnected)(&fCableConnected); H();
1161 rc = CFGMR3InsertInteger(pCfg, "CableConnected", fCableConnected ? 1 : 0); RC_CHECK();
1162
1163 /*
1164 * Line speed to report from custom drivers
1165 */
1166 ULONG ulLineSpeed;
1167 hrc = networkAdapter->COMGETTER(LineSpeed)(&ulLineSpeed); H();
1168 rc = CFGMR3InsertInteger(pCfg, "LineSpeed", ulLineSpeed); RC_CHECK();
1169
1170 /*
1171 * Attach the status driver.
1172 */
1173 rc = CFGMR3InsertNode(pInst, "LUN#999", &pLunL0); RC_CHECK();
1174 rc = CFGMR3InsertString(pLunL0, "Driver", "MainStatus"); RC_CHECK();
1175 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1176 rc = CFGMR3InsertInteger(pCfg, "papLeds", (uintptr_t)&pConsole->mapNetworkLeds[ulInstance]); RC_CHECK();
1177
1178 /*
1179 * Configure the network card now
1180 */
1181 rc = configNetwork(pConsole, pszAdapterName, ulInstance, 0, networkAdapter,
1182 pCfg, pLunL0, pInst, false /*fAttachDetach*/); RC_CHECK();
1183 }
1184
1185 /*
1186 * Serial (UART) Ports
1187 */
1188 rc = CFGMR3InsertNode(pDevices, "serial", &pDev); RC_CHECK();
1189 for (ULONG ulInstance = 0; ulInstance < SchemaDefs::SerialPortCount; ++ulInstance)
1190 {
1191 ComPtr<ISerialPort> serialPort;
1192 hrc = pMachine->GetSerialPort (ulInstance, serialPort.asOutParam()); H();
1193 BOOL fEnabled = FALSE;
1194 if (serialPort)
1195 hrc = serialPort->COMGETTER(Enabled)(&fEnabled); H();
1196 if (!fEnabled)
1197 continue;
1198
1199 rc = CFGMR3InsertNodeF(pDev, &pInst, "%u", ulInstance); RC_CHECK();
1200 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1201
1202 ULONG ulIRQ;
1203 hrc = serialPort->COMGETTER(IRQ)(&ulIRQ); H();
1204 rc = CFGMR3InsertInteger(pCfg, "IRQ", ulIRQ); RC_CHECK();
1205 ULONG ulIOBase;
1206 hrc = serialPort->COMGETTER(IOBase)(&ulIOBase); H();
1207 rc = CFGMR3InsertInteger(pCfg, "IOBase", ulIOBase); RC_CHECK();
1208 BOOL fServer;
1209 hrc = serialPort->COMGETTER(Server)(&fServer); H();
1210 hrc = serialPort->COMGETTER(Path)(&str); H();
1211 PortMode_T eHostMode;
1212 hrc = serialPort->COMGETTER(HostMode)(&eHostMode); H();
1213 if (eHostMode != PortMode_Disconnected)
1214 {
1215 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
1216 if (eHostMode == PortMode_HostPipe)
1217 {
1218 rc = CFGMR3InsertString(pLunL0, "Driver", "Char"); RC_CHECK();
1219 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL1); RC_CHECK();
1220 rc = CFGMR3InsertString(pLunL1, "Driver", "NamedPipe"); RC_CHECK();
1221 rc = CFGMR3InsertNode(pLunL1, "Config", &pLunL2); RC_CHECK();
1222 rc = CFGMR3InsertStringW(pLunL2, "Location", str); RC_CHECK();
1223 rc = CFGMR3InsertInteger(pLunL2, "IsServer", fServer); RC_CHECK();
1224 }
1225 else if (eHostMode == PortMode_HostDevice)
1226 {
1227 rc = CFGMR3InsertString(pLunL0, "Driver", "Host Serial"); RC_CHECK();
1228 rc = CFGMR3InsertNode(pLunL0, "Config", &pLunL1); RC_CHECK();
1229 rc = CFGMR3InsertStringW(pLunL1, "DevicePath", str); RC_CHECK();
1230 }
1231 else if (eHostMode == PortMode_RawFile)
1232 {
1233 rc = CFGMR3InsertString(pLunL0, "Driver", "Char"); RC_CHECK();
1234 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL1); RC_CHECK();
1235 rc = CFGMR3InsertString(pLunL1, "Driver", "RawFile"); RC_CHECK();
1236 rc = CFGMR3InsertNode(pLunL1, "Config", &pLunL2); RC_CHECK();
1237 rc = CFGMR3InsertStringW(pLunL2, "Location", str); RC_CHECK();
1238 }
1239 }
1240 STR_FREE();
1241 }
1242
1243 /*
1244 * Parallel (LPT) Ports
1245 */
1246 rc = CFGMR3InsertNode(pDevices, "parallel", &pDev); RC_CHECK();
1247 for (ULONG ulInstance = 0; ulInstance < SchemaDefs::ParallelPortCount; ++ulInstance)
1248 {
1249 ComPtr<IParallelPort> parallelPort;
1250 hrc = pMachine->GetParallelPort(ulInstance, parallelPort.asOutParam()); H();
1251 BOOL fEnabled = FALSE;
1252 if (parallelPort)
1253 {
1254 hrc = parallelPort->COMGETTER(Enabled)(&fEnabled); H();
1255 }
1256 if (!fEnabled)
1257 continue;
1258
1259 rc = CFGMR3InsertNodeF(pDev, &pInst, "%u", ulInstance); RC_CHECK();
1260 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1261
1262 ULONG ulIRQ;
1263 hrc = parallelPort->COMGETTER(IRQ)(&ulIRQ); H();
1264 rc = CFGMR3InsertInteger(pCfg, "IRQ", ulIRQ); RC_CHECK();
1265 ULONG ulIOBase;
1266 hrc = parallelPort->COMGETTER(IOBase)(&ulIOBase); H();
1267 rc = CFGMR3InsertInteger(pCfg, "IOBase", ulIOBase); RC_CHECK();
1268 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
1269 rc = CFGMR3InsertString(pLunL0, "Driver", "HostParallel"); RC_CHECK();
1270 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL1); RC_CHECK();
1271 hrc = parallelPort->COMGETTER(Path)(&str); H();
1272 rc = CFGMR3InsertStringW(pLunL1, "DevicePath", str); RC_CHECK();
1273 STR_FREE();
1274 }
1275
1276 /*
1277 * VMM Device
1278 */
1279 rc = CFGMR3InsertNode(pDevices, "VMMDev", &pDev); RC_CHECK();
1280 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
1281 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1282 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
1283 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", 4); RC_CHECK();
1284 Assert(!afPciDeviceNo[4]);
1285 afPciDeviceNo[4] = true;
1286 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
1287 Bstr hwVersion;
1288 hrc = pMachine->COMGETTER(HardwareVersion)(hwVersion.asOutParam()); H();
1289 if (hwVersion.compare(Bstr("1")) == 0) /* <= 2.0.x */
1290 {
1291 CFGMR3InsertInteger(pCfg, "HeapEnabled", 0); RC_CHECK();
1292 }
1293
1294 /* the VMM device's Main driver */
1295 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
1296 rc = CFGMR3InsertString(pLunL0, "Driver", "HGCM"); RC_CHECK();
1297 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1298 VMMDev *pVMMDev = pConsole->mVMMDev;
1299 rc = CFGMR3InsertInteger(pCfg, "Object", (uintptr_t)pVMMDev); RC_CHECK();
1300
1301 /*
1302 * Attach the status driver.
1303 */
1304 rc = CFGMR3InsertNode(pInst, "LUN#999", &pLunL0); RC_CHECK();
1305 rc = CFGMR3InsertString(pLunL0, "Driver", "MainStatus"); RC_CHECK();
1306 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1307 rc = CFGMR3InsertInteger(pCfg, "papLeds", (uintptr_t)&pConsole->mapSharedFolderLed); RC_CHECK();
1308 rc = CFGMR3InsertInteger(pCfg, "First", 0); RC_CHECK();
1309 rc = CFGMR3InsertInteger(pCfg, "Last", 0); RC_CHECK();
1310
1311 /*
1312 * Audio Sniffer Device
1313 */
1314 rc = CFGMR3InsertNode(pDevices, "AudioSniffer", &pDev); RC_CHECK();
1315 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
1316 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1317
1318 /* the Audio Sniffer device's Main driver */
1319 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
1320 rc = CFGMR3InsertString(pLunL0, "Driver", "MainAudioSniffer"); RC_CHECK();
1321 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1322 AudioSniffer *pAudioSniffer = pConsole->mAudioSniffer;
1323 rc = CFGMR3InsertInteger(pCfg, "Object", (uintptr_t)pAudioSniffer); RC_CHECK();
1324
1325 /*
1326 * AC'97 ICH / SoundBlaster16 audio
1327 */
1328 BOOL enabled;
1329 ComPtr<IAudioAdapter> audioAdapter;
1330 hrc = pMachine->COMGETTER(AudioAdapter)(audioAdapter.asOutParam()); H();
1331 if (audioAdapter)
1332 hrc = audioAdapter->COMGETTER(Enabled)(&enabled); H();
1333
1334 if (enabled)
1335 {
1336 AudioControllerType_T audioController;
1337 hrc = audioAdapter->COMGETTER(AudioController)(&audioController); H();
1338 switch (audioController)
1339 {
1340 case AudioControllerType_AC97:
1341 {
1342 /* default: ICH AC97 */
1343 rc = CFGMR3InsertNode(pDevices, "ichac97", &pDev); RC_CHECK();
1344 rc = CFGMR3InsertNode(pDev, "0", &pInst);
1345 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* bool */ RC_CHECK();
1346 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", 5); RC_CHECK();
1347 Assert(!afPciDeviceNo[5]);
1348 afPciDeviceNo[5] = true;
1349 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
1350 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1351 break;
1352 }
1353 case AudioControllerType_SB16:
1354 {
1355 /* legacy SoundBlaster16 */
1356 rc = CFGMR3InsertNode(pDevices, "sb16", &pDev); RC_CHECK();
1357 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
1358 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* bool */ RC_CHECK();
1359 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1360 rc = CFGMR3InsertInteger(pCfg, "IRQ", 5); RC_CHECK();
1361 rc = CFGMR3InsertInteger(pCfg, "DMA", 1); RC_CHECK();
1362 rc = CFGMR3InsertInteger(pCfg, "DMA16", 5); RC_CHECK();
1363 rc = CFGMR3InsertInteger(pCfg, "Port", 0x220); RC_CHECK();
1364 rc = CFGMR3InsertInteger(pCfg, "Version", 0x0405); RC_CHECK();
1365 break;
1366 }
1367 }
1368
1369 /* the Audio driver */
1370 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
1371 rc = CFGMR3InsertString(pLunL0, "Driver", "AUDIO"); RC_CHECK();
1372 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1373
1374 AudioDriverType_T audioDriver;
1375 hrc = audioAdapter->COMGETTER(AudioDriver)(&audioDriver); H();
1376 switch (audioDriver)
1377 {
1378 case AudioDriverType_Null:
1379 {
1380 rc = CFGMR3InsertString(pCfg, "AudioDriver", "null"); RC_CHECK();
1381 break;
1382 }
1383#ifdef RT_OS_WINDOWS
1384#ifdef VBOX_WITH_WINMM
1385 case AudioDriverType_WinMM:
1386 {
1387 rc = CFGMR3InsertString(pCfg, "AudioDriver", "winmm"); RC_CHECK();
1388 break;
1389 }
1390#endif
1391 case AudioDriverType_DirectSound:
1392 {
1393 rc = CFGMR3InsertString(pCfg, "AudioDriver", "dsound"); RC_CHECK();
1394 break;
1395 }
1396#endif /* RT_OS_WINDOWS */
1397#ifdef RT_OS_SOLARIS
1398 case AudioDriverType_SolAudio:
1399 {
1400 rc = CFGMR3InsertString(pCfg, "AudioDriver", "solaudio"); RC_CHECK();
1401 break;
1402 }
1403#endif
1404#ifdef RT_OS_LINUX
1405# ifdef VBOX_WITH_ALSA
1406 case AudioDriverType_ALSA:
1407 {
1408 rc = CFGMR3InsertString(pCfg, "AudioDriver", "alsa"); RC_CHECK();
1409 break;
1410 }
1411# endif
1412# ifdef VBOX_WITH_PULSE
1413 case AudioDriverType_Pulse:
1414 {
1415 rc = CFGMR3InsertString(pCfg, "AudioDriver", "pulse"); RC_CHECK();
1416 break;
1417 }
1418# endif
1419#endif /* RT_OS_LINUX */
1420#if defined (RT_OS_LINUX) || defined (RT_OS_FREEBSD) || defined(VBOX_WITH_SOLARIS_OSS)
1421 case AudioDriverType_OSS:
1422 {
1423 rc = CFGMR3InsertString(pCfg, "AudioDriver", "oss"); RC_CHECK();
1424 break;
1425 }
1426#endif
1427#ifdef RT_OS_DARWIN
1428 case AudioDriverType_CoreAudio:
1429 {
1430 rc = CFGMR3InsertString(pCfg, "AudioDriver", "coreaudio"); RC_CHECK();
1431 break;
1432 }
1433#endif
1434 }
1435 hrc = pMachine->COMGETTER(Name)(&str); H();
1436 rc = CFGMR3InsertStringW(pCfg, "StreamName", str); RC_CHECK();
1437 STR_FREE();
1438 }
1439
1440 /*
1441 * The USB Controller.
1442 */
1443 ComPtr<IUSBController> USBCtlPtr;
1444 hrc = pMachine->COMGETTER(USBController)(USBCtlPtr.asOutParam());
1445 if (USBCtlPtr)
1446 {
1447 BOOL fEnabled;
1448 hrc = USBCtlPtr->COMGETTER(Enabled)(&fEnabled); H();
1449 if (fEnabled)
1450 {
1451 rc = CFGMR3InsertNode(pDevices, "usb-ohci", &pDev); RC_CHECK();
1452 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
1453 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1454 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
1455 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", 6); RC_CHECK();
1456 Assert(!afPciDeviceNo[6]);
1457 afPciDeviceNo[6] = true;
1458 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
1459
1460 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
1461 rc = CFGMR3InsertString(pLunL0, "Driver", "VUSBRootHub"); RC_CHECK();
1462 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1463
1464 /*
1465 * Attach the status driver.
1466 */
1467 rc = CFGMR3InsertNode(pInst, "LUN#999", &pLunL0); RC_CHECK();
1468 rc = CFGMR3InsertString(pLunL0, "Driver", "MainStatus"); RC_CHECK();
1469 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1470 rc = CFGMR3InsertInteger(pCfg, "papLeds", (uintptr_t)&pConsole->mapUSBLed[0]);RC_CHECK();
1471 rc = CFGMR3InsertInteger(pCfg, "First", 0); RC_CHECK();
1472 rc = CFGMR3InsertInteger(pCfg, "Last", 0); RC_CHECK();
1473
1474#ifdef VBOX_WITH_EHCI
1475 hrc = USBCtlPtr->COMGETTER(EnabledEhci)(&fEnabled); H();
1476 if (fEnabled)
1477 {
1478 rc = CFGMR3InsertNode(pDevices, "usb-ehci", &pDev); RC_CHECK();
1479 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
1480 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1481 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* bool */ RC_CHECK();
1482 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", 11); RC_CHECK();
1483 Assert(!afPciDeviceNo[11]);
1484 afPciDeviceNo[11] = true;
1485 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
1486
1487 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
1488 rc = CFGMR3InsertString(pLunL0, "Driver", "VUSBRootHub"); RC_CHECK();
1489 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1490
1491 /*
1492 * Attach the status driver.
1493 */
1494 rc = CFGMR3InsertNode(pInst, "LUN#999", &pLunL0); RC_CHECK();
1495 rc = CFGMR3InsertString(pLunL0, "Driver", "MainStatus"); RC_CHECK();
1496 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1497 rc = CFGMR3InsertInteger(pCfg, "papLeds", (uintptr_t)&pConsole->mapUSBLed[1]);RC_CHECK();
1498 rc = CFGMR3InsertInteger(pCfg, "First", 0); RC_CHECK();
1499 rc = CFGMR3InsertInteger(pCfg, "Last", 0); RC_CHECK();
1500 }
1501 else
1502#endif
1503 {
1504 /*
1505 * Global USB options, currently unused as we'll apply the 2.0 -> 1.1 morphing
1506 * on a per device level now.
1507 */
1508 rc = CFGMR3InsertNode(pRoot, "USB", &pCfg); RC_CHECK();
1509 rc = CFGMR3InsertNode(pCfg, "USBProxy", &pCfg); RC_CHECK();
1510 rc = CFGMR3InsertNode(pCfg, "GlobalConfig", &pCfg); RC_CHECK();
1511 // This globally enables the 2.0 -> 1.1 device morphing of proxied devies to keep windows quiet.
1512 //rc = CFGMR3InsertInteger(pCfg, "Force11Device", true); RC_CHECK();
1513 // The following breaks stuff, but it makes MSDs work in vista. (I include it here so
1514 // that it's documented somewhere.) Users needing it can use:
1515 // VBoxManage setextradata "myvm" "VBoxInternal/USB/USBProxy/GlobalConfig/Force11PacketSize" 1
1516 //rc = CFGMR3InsertInteger(pCfg, "Force11PacketSize", true); RC_CHECK();
1517 }
1518 }
1519 }
1520
1521 /*
1522 * Clipboard
1523 */
1524 {
1525 ClipboardMode_T mode = ClipboardMode_Disabled;
1526 hrc = pMachine->COMGETTER(ClipboardMode)(&mode); H();
1527
1528 if (mode != ClipboardMode_Disabled)
1529 {
1530 /* Load the service */
1531 rc = pConsole->mVMMDev->hgcmLoadService ("VBoxSharedClipboard", "VBoxSharedClipboard");
1532
1533 if (RT_FAILURE(rc))
1534 {
1535 LogRel(("VBoxSharedClipboard is not available. rc = %Rrc\n", rc));
1536 /* That is not a fatal failure. */
1537 rc = VINF_SUCCESS;
1538 }
1539 else
1540 {
1541 /* Setup the service. */
1542 VBOXHGCMSVCPARM parm;
1543
1544 parm.type = VBOX_HGCM_SVC_PARM_32BIT;
1545
1546 switch (mode)
1547 {
1548 default:
1549 case ClipboardMode_Disabled:
1550 {
1551 LogRel(("VBoxSharedClipboard mode: Off\n"));
1552 parm.u.uint32 = VBOX_SHARED_CLIPBOARD_MODE_OFF;
1553 break;
1554 }
1555 case ClipboardMode_GuestToHost:
1556 {
1557 LogRel(("VBoxSharedClipboard mode: Guest to Host\n"));
1558 parm.u.uint32 = VBOX_SHARED_CLIPBOARD_MODE_GUEST_TO_HOST;
1559 break;
1560 }
1561 case ClipboardMode_HostToGuest:
1562 {
1563 LogRel(("VBoxSharedClipboard mode: Host to Guest\n"));
1564 parm.u.uint32 = VBOX_SHARED_CLIPBOARD_MODE_HOST_TO_GUEST;
1565 break;
1566 }
1567 case ClipboardMode_Bidirectional:
1568 {
1569 LogRel(("VBoxSharedClipboard mode: Bidirectional\n"));
1570 parm.u.uint32 = VBOX_SHARED_CLIPBOARD_MODE_BIDIRECTIONAL;
1571 break;
1572 }
1573 }
1574
1575 pConsole->mVMMDev->hgcmHostCall ("VBoxSharedClipboard", VBOX_SHARED_CLIPBOARD_HOST_FN_SET_MODE, 1, &parm);
1576
1577 Log(("Set VBoxSharedClipboard mode\n"));
1578 }
1579 }
1580 }
1581
1582#ifdef VBOX_WITH_CROGL
1583 /*
1584 * crOpenGL
1585 */
1586 {
1587 BOOL fEnabled = false;
1588 hrc = pMachine->COMGETTER(Accelerate3DEnabled)(&fEnabled); H();
1589
1590 if (fEnabled)
1591 {
1592 /* Load the service */
1593 rc = pConsole->mVMMDev->hgcmLoadService ("VBoxSharedCrOpenGL", "VBoxSharedCrOpenGL");
1594 if (RT_FAILURE(rc))
1595 {
1596 LogRel(("Failed to load Shared OpenGL service %Rrc\n", rc));
1597 /* That is not a fatal failure. */
1598 rc = VINF_SUCCESS;
1599 }
1600 else
1601 {
1602 LogRel(("Shared crOpenGL service loaded.\n"));
1603
1604 /* Setup the service. */
1605 VBOXHGCMSVCPARM parm;
1606 parm.type = VBOX_HGCM_SVC_PARM_PTR;
1607
1608 parm.u.pointer.addr = pConsole->getDisplay()->getFramebuffer();
1609 parm.u.pointer.size = sizeof(IFramebuffer *);
1610
1611 rc = pConsole->mVMMDev->hgcmHostCall("VBoxSharedCrOpenGL", SHCRGL_HOST_FN_SET_FRAMEBUFFER, 1, &parm);
1612 if (!RT_SUCCESS(rc))
1613 AssertMsgFailed(("SHCRGL_HOST_FN_SET_FRAMEBUFFER failed with %Rrc\n", rc));
1614
1615 parm.u.pointer.addr = pVM;
1616 parm.u.pointer.size = sizeof(pVM);
1617 rc = pConsole->mVMMDev->hgcmHostCall("VBoxSharedCrOpenGL", SHCRGL_HOST_FN_SET_VM, 1, &parm);
1618 if (!RT_SUCCESS(rc))
1619 AssertMsgFailed(("SHCRGL_HOST_FN_SET_VM failed with %Rrc\n", rc));
1620 }
1621
1622 }
1623 }
1624#endif
1625
1626#ifdef VBOX_WITH_GUEST_PROPS
1627 /*
1628 * Guest property service
1629 */
1630
1631 rc = configGuestProperties(pConsole);
1632#endif /* VBOX_WITH_GUEST_PROPS defined */
1633
1634 /*
1635 * CFGM overlay handling.
1636 *
1637 * Here we check the extra data entries for CFGM values
1638 * and create the nodes and insert the values on the fly. Existing
1639 * values will be removed and reinserted. CFGM is typed, so by default
1640 * we will guess whether it's a string or an integer (byte arrays are
1641 * not currently supported). It's possible to override this autodetection
1642 * by adding "string:", "integer:" or "bytes:" (future).
1643 *
1644 * We first perform a run on global extra data, then on the machine
1645 * extra data to support global settings with local overrides.
1646 *
1647 */
1648 /** @todo add support for removing nodes and byte blobs. */
1649 SafeArray<BSTR> aGlobalExtraDataKeys;
1650 SafeArray<BSTR> aMachineExtraDataKeys;
1651 /*
1652 * Get the next key
1653 */
1654 if (FAILED(hrc = virtualBox->GetExtraDataKeys(ComSafeArrayAsOutParam(aGlobalExtraDataKeys))))
1655 AssertMsgFailed(("VirtualBox::GetExtraDataKeys failed with %Rrc\n", hrc));
1656
1657 // remember the no. of global values so we can call the correct method below
1658 size_t cGlobalValues = aGlobalExtraDataKeys.size();
1659
1660 if (FAILED(hrc = pMachine->GetExtraDataKeys(ComSafeArrayAsOutParam(aMachineExtraDataKeys))))
1661 AssertMsgFailed(("IMachine::GetExtraDataKeys failed with %Rrc\n", hrc));
1662
1663 // build a combined list from global keys...
1664 std::list<Utf8Str> llExtraDataKeys;
1665 for (size_t i = 0; i < aGlobalExtraDataKeys.size(); ++i)
1666 llExtraDataKeys.push_back(Utf8Str(aGlobalExtraDataKeys[i]));
1667 // ... and machine keys
1668 for (size_t i = 0; i < aMachineExtraDataKeys.size(); ++i)
1669 llExtraDataKeys.push_back(Utf8Str(aMachineExtraDataKeys[i]));
1670
1671 size_t i = 0;
1672 for (std::list<Utf8Str>::const_iterator it = llExtraDataKeys.begin();
1673 it != llExtraDataKeys.end();
1674 ++it, ++i)
1675 {
1676 const Utf8Str &strKey = *it;
1677
1678 /*
1679 * We only care about keys starting with "VBoxInternal/" (skip "G:" or "M:")
1680 */
1681 if (!strKey.startsWith("VBoxInternal/"))
1682 continue;
1683
1684 const char *pszExtraDataKey = strKey.raw() + sizeof("VBoxInternal/") - 1;
1685
1686 // get the value
1687 Bstr strExtraDataValue;
1688 if (i < cGlobalValues)
1689 // this is still one of the global values:
1690 hrc = virtualBox->GetExtraData(Bstr(strKey), strExtraDataValue.asOutParam());
1691 else
1692 hrc = pMachine->GetExtraData(Bstr(strKey), strExtraDataValue.asOutParam());
1693 if (FAILED(hrc))
1694 LogRel(("Warning: Cannot get extra data key %s, rc = %Rrc\n", strKey.raw(), hrc));
1695
1696 /*
1697 * The key will be in the format "Node1/Node2/Value" or simply "Value".
1698 * Split the two and get the node, delete the value and create the node
1699 * if necessary.
1700 */
1701 PCFGMNODE pNode;
1702 const char *pszCFGMValueName = strrchr(pszExtraDataKey, '/');
1703 if (pszCFGMValueName)
1704 {
1705 /* terminate the node and advance to the value (Utf8Str might not
1706 offically like this but wtf) */
1707 *(char*)pszCFGMValueName = '\0';
1708 ++pszCFGMValueName;
1709
1710 /* does the node already exist? */
1711 pNode = CFGMR3GetChild(pRoot, pszExtraDataKey);
1712 if (pNode)
1713 CFGMR3RemoveValue(pNode, pszCFGMValueName);
1714 else
1715 {
1716 /* create the node */
1717 rc = CFGMR3InsertNode(pRoot, pszExtraDataKey, &pNode);
1718 if (RT_FAILURE(rc))
1719 {
1720 AssertLogRelMsgRC(rc, ("failed to insert node '%s'\n", pszExtraDataKey));
1721 continue;
1722 }
1723 Assert(pNode);
1724 }
1725 }
1726 else
1727 {
1728 /* root value (no node path). */
1729 pNode = pRoot;
1730 pszCFGMValueName = pszExtraDataKey;
1731 pszExtraDataKey--;
1732 CFGMR3RemoveValue(pNode, pszCFGMValueName);
1733 }
1734
1735 /*
1736 * Now let's have a look at the value.
1737 * Empty strings means that we should remove the value, which we've
1738 * already done above.
1739 */
1740 Utf8Str strCFGMValueUtf8(strExtraDataValue);
1741 const char *pszCFGMValue = strCFGMValueUtf8.raw();
1742 if ( pszCFGMValue
1743 && *pszCFGMValue)
1744 {
1745 uint64_t u64Value;
1746
1747 /* check for type prefix first. */
1748 if (!strncmp(pszCFGMValue, "string:", sizeof("string:") - 1))
1749 rc = CFGMR3InsertString(pNode, pszCFGMValueName, pszCFGMValue + sizeof("string:") - 1);
1750 else if (!strncmp(pszCFGMValue, "integer:", sizeof("integer:") - 1))
1751 {
1752 rc = RTStrToUInt64Full(pszCFGMValue + sizeof("integer:") - 1, 0, &u64Value);
1753 if (RT_SUCCESS(rc))
1754 rc = CFGMR3InsertInteger(pNode, pszCFGMValueName, u64Value);
1755 }
1756 else if (!strncmp(pszCFGMValue, "bytes:", sizeof("bytes:") - 1))
1757 rc = VERR_NOT_IMPLEMENTED;
1758 /* auto detect type. */
1759 else if (RT_SUCCESS(RTStrToUInt64Full(pszCFGMValue, 0, &u64Value)))
1760 rc = CFGMR3InsertInteger(pNode, pszCFGMValueName, u64Value);
1761 else
1762 rc = CFGMR3InsertString(pNode, pszCFGMValueName, pszCFGMValue);
1763 AssertLogRelMsgRC(rc, ("failed to insert CFGM value '%s' to key '%s'\n", pszCFGMValue, pszExtraDataKey));
1764 }
1765 }
1766
1767 /*
1768 * ACPI
1769 */
1770 BOOL fACPI;
1771 hrc = biosSettings->COMGETTER(ACPIEnabled)(&fACPI); H();
1772 if (fACPI)
1773 {
1774 BOOL fShowCpu = fExtProfile;
1775 /* Always show the CPU leafs when we have multiple VCPUs or when the IO-APIC is enabled.
1776 * The Windows SMP kernel needs a CPU leaf or else its idle loop will burn cpu cycles; the
1777 * intelppm driver refuses to register an idle state handler.
1778 */
1779 if ((cCpus > 1) || fIOAPIC)
1780 fShowCpu = true;
1781
1782 rc = CFGMR3InsertNode(pDevices, "acpi", &pDev); RC_CHECK();
1783 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
1784 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
1785 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1786 rc = CFGMR3InsertInteger(pCfg, "RamSize", cbRam); RC_CHECK();
1787 rc = CFGMR3InsertInteger(pCfg, "RamHoleSize", cbRamHole); RC_CHECK();
1788 rc = CFGMR3InsertInteger(pCfg, "NumCPUs", cCpus); RC_CHECK();
1789
1790 rc = CFGMR3InsertInteger(pCfg, "IOAPIC", fIOAPIC); RC_CHECK();
1791 rc = CFGMR3InsertInteger(pCfg, "FdcEnabled", fFdcEnabled); RC_CHECK();
1792#ifdef VBOX_WITH_HPET
1793 rc = CFGMR3InsertInteger(pCfg, "HpetEnabled", fHpetEnabled); RC_CHECK();
1794#endif
1795#ifdef VBOX_WITH_SMC
1796 rc = CFGMR3InsertInteger(pCfg, "SmcEnabled", fSmcEnabled); RC_CHECK();
1797#endif
1798 rc = CFGMR3InsertInteger(pCfg, "ShowRtc", fExtProfile); RC_CHECK();
1799
1800 rc = CFGMR3InsertInteger(pCfg, "ShowCpu", fShowCpu); RC_CHECK();
1801 rc = CFGMR3InsertInteger(pInst, "PCIDeviceNo", 7); RC_CHECK();
1802 Assert(!afPciDeviceNo[7]);
1803 afPciDeviceNo[7] = true;
1804 rc = CFGMR3InsertInteger(pInst, "PCIFunctionNo", 0); RC_CHECK();
1805
1806 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
1807 rc = CFGMR3InsertString(pLunL0, "Driver", "ACPIHost"); RC_CHECK();
1808 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
1809 }
1810
1811 /*
1812 * Firmware.
1813 */
1814#ifdef VBOX_WITH_EFI
1815 Bstr tmpStr1;
1816 hrc = pMachine->GetExtraData(Bstr("VBoxInternal2/UseEFI"), tmpStr1.asOutParam()); H();
1817 BOOL fEfiEnabled = !tmpStr1.isEmpty();
1818
1819 /**
1820 * @todo: VBoxInternal2/UseEFI extradata will go away soon, and we'll
1821 * just use this code
1822 */
1823 if (!fEfiEnabled)
1824 {
1825 FirmwareType_T eType = FirmwareType_BIOS;
1826 hrc = pMachine->COMGETTER(FirmwareType)(&eType); H();
1827 fEfiEnabled = (eType == FirmwareType_EFI);
1828 }
1829#else
1830 BOOL fEfiEnabled = false;
1831#endif
1832 if (!fEfiEnabled)
1833 {
1834 /*
1835 * PC Bios.
1836 */
1837 rc = CFGMR3InsertNode(pDevices, "pcbios", &pDev); RC_CHECK();
1838 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
1839 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
1840 rc = CFGMR3InsertNode(pInst, "Config", &pBiosCfg); RC_CHECK();
1841 rc = CFGMR3InsertInteger(pBiosCfg, "RamSize", cbRam); RC_CHECK();
1842 rc = CFGMR3InsertInteger(pBiosCfg, "RamHoleSize", cbRamHole); RC_CHECK();
1843 rc = CFGMR3InsertInteger(pBiosCfg, "NumCPUs", cCpus); RC_CHECK();
1844 rc = CFGMR3InsertString(pBiosCfg, "HardDiskDevice", "piix3ide"); RC_CHECK();
1845 rc = CFGMR3InsertString(pBiosCfg, "FloppyDevice", "i82078"); RC_CHECK();
1846 rc = CFGMR3InsertInteger(pBiosCfg, "IOAPIC", fIOAPIC); RC_CHECK();
1847 rc = CFGMR3InsertInteger(pBiosCfg, "PXEDebug", fPXEDebug); RC_CHECK();
1848 rc = CFGMR3InsertBytes(pBiosCfg, "UUID", pUuid, sizeof(*pUuid)); RC_CHECK();
1849
1850 DeviceType_T bootDevice;
1851 if (SchemaDefs::MaxBootPosition > 9)
1852 {
1853 AssertMsgFailed (("Too many boot devices %d\n",
1854 SchemaDefs::MaxBootPosition));
1855 return VERR_INVALID_PARAMETER;
1856 }
1857
1858 for (ULONG pos = 1; pos <= SchemaDefs::MaxBootPosition; ++pos)
1859 {
1860 hrc = pMachine->GetBootOrder(pos, &bootDevice); H();
1861
1862 char szParamName[] = "BootDeviceX";
1863 szParamName[sizeof (szParamName) - 2] = ((char (pos - 1)) + '0');
1864
1865 const char *pszBootDevice;
1866 switch (bootDevice)
1867 {
1868 case DeviceType_Null:
1869 pszBootDevice = "NONE";
1870 break;
1871 case DeviceType_HardDisk:
1872 pszBootDevice = "IDE";
1873 break;
1874 case DeviceType_DVD:
1875 pszBootDevice = "DVD";
1876 break;
1877 case DeviceType_Floppy:
1878 pszBootDevice = "FLOPPY";
1879 break;
1880 case DeviceType_Network:
1881 pszBootDevice = "LAN";
1882 break;
1883 default:
1884 AssertMsgFailed(("Invalid bootDevice=%d\n", bootDevice));
1885 return VMSetError(pVM, VERR_INVALID_PARAMETER, RT_SRC_POS,
1886 N_("Invalid boot device '%d'"), bootDevice);
1887 }
1888 rc = CFGMR3InsertString(pBiosCfg, szParamName, pszBootDevice); RC_CHECK();
1889 }
1890 }
1891 else
1892 {
1893 /*
1894 * EFI.
1895 */
1896 rc = CFGMR3InsertNode(pDevices, "efi", &pDev); RC_CHECK();
1897 rc = CFGMR3InsertNode(pDev, "0", &pInst); RC_CHECK();
1898 rc = CFGMR3InsertInteger(pInst, "Trusted", 1); /* boolean */ RC_CHECK();
1899 rc = CFGMR3InsertNode(pInst, "Config", &pCfg); RC_CHECK();
1900 rc = CFGMR3InsertInteger(pCfg, "RamSize", cbRam); RC_CHECK();
1901 rc = CFGMR3InsertInteger(pCfg, "RamHoleSize", cbRamHole); RC_CHECK();
1902 rc = CFGMR3InsertInteger(pCfg, "NumCPUs", cCpus); RC_CHECK();
1903 }
1904
1905#undef STR_FREE
1906#undef H
1907#undef RC_CHECK
1908
1909 /* Register VM state change handler */
1910 int rc2 = VMR3AtStateRegister (pVM, Console::vmstateChangeCallback, pConsole);
1911 AssertRC (rc2);
1912 if (RT_SUCCESS(rc))
1913 rc = rc2;
1914
1915 /* Register VM runtime error handler */
1916 rc2 = VMR3AtRuntimeErrorRegister (pVM, Console::setVMRuntimeErrorCallback, pConsole);
1917 AssertRC (rc2);
1918 if (RT_SUCCESS(rc))
1919 rc = rc2;
1920
1921 LogFlowFunc (("vrc = %Rrc\n", rc));
1922 LogFlowFuncLeave();
1923
1924 return rc;
1925}
1926
1927
1928/**
1929 * Construct the Network configuration tree
1930 *
1931 * @returns VBox status code.
1932 *
1933 * @param pThis Pointer to the Console object.
1934 * @param pszDevice The PDM device name.
1935 * @param uInstance The PDM device instance.
1936 * @param uLun The PDM LUN number of the drive.
1937 * @param aNetworkAdapter The network adapter whose attachment needs to be changed
1938 * @param pCfg Configuration node for the device
1939 * @param pLunL0 To store the pointer to the LUN#0.
1940 * @param pInst The instance CFGM node
1941 * @param fAttachDetach To determine if the network attachment should
1942 * be attached/detached after/before
1943 * configuration.
1944 *
1945 * @note Locks the Console object for writing.
1946 */
1947/*static*/ int Console::configNetwork(Console *pThis, const char *pszDevice,
1948 unsigned uInstance, unsigned uLun,
1949 INetworkAdapter *aNetworkAdapter,
1950 PCFGMNODE pCfg, PCFGMNODE pLunL0,
1951 PCFGMNODE pInst, bool fAttachDetach)
1952{
1953 int rc = VINF_SUCCESS;
1954
1955 AutoCaller autoCaller(pThis);
1956 AssertComRCReturn(autoCaller.rc(), VERR_ACCESS_DENIED);
1957
1958 /*
1959 * Locking the object before doing VMR3* calls is quite safe here, since
1960 * we're on EMT. Write lock is necessary because we indirectly modify the
1961 * meAttachmentType member.
1962 */
1963 AutoWriteLock alock(pThis);
1964
1965 PVM pVM = pThis->mpVM;
1966 BSTR str = NULL;
1967
1968#define STR_FREE() do { if (str) { SysFreeString(str); str = NULL; } } while (0)
1969#define RC_CHECK() do { if (RT_FAILURE(rc)) { AssertMsgFailed(("rc=%Rrc\n", rc)); STR_FREE(); return rc; } } while (0)
1970#define H() do { if (FAILED(hrc)) { AssertMsgFailed(("hrc=%#x\n", hrc)); STR_FREE(); return VERR_GENERAL_FAILURE; } } while (0)
1971
1972 HRESULT hrc;
1973 ComPtr<IMachine> pMachine = pThis->machine();
1974
1975 ComPtr<IVirtualBox> virtualBox;
1976 hrc = pMachine->COMGETTER(Parent)(virtualBox.asOutParam());
1977 H();
1978
1979 ComPtr<IHost> host;
1980 hrc = virtualBox->COMGETTER(Host)(host.asOutParam());
1981 H();
1982
1983 BOOL fSniffer;
1984 hrc = aNetworkAdapter->COMGETTER(TraceEnabled)(&fSniffer);
1985 H();
1986
1987 if (fAttachDetach && fSniffer)
1988 {
1989 const char *pszNetDriver = "IntNet";
1990 if (pThis->meAttachmentType[uInstance] == NetworkAttachmentType_NAT)
1991 pszNetDriver = "NAT";
1992#if !defined(VBOX_WITH_NETFLT) && defined(RT_OS_LINUX)
1993 if (pThis->meAttachmentType[uInstance] == NetworkAttachmentType_Bridged)
1994 pszNetDriver = "HostInterface";
1995#endif
1996
1997 rc = PDMR3DriverDetach(pVM, pszDevice, uInstance, uLun, pszNetDriver, 0, 0 /*fFlags*/);
1998 if (rc == VINF_PDM_NO_DRIVER_ATTACHED_TO_LUN)
1999 rc = VINF_SUCCESS;
2000 AssertLogRelRCReturn(rc, rc);
2001
2002 pLunL0 = CFGMR3GetChildF(pInst, "LUN#%u", uLun);
2003 PCFGMNODE pLunAD = CFGMR3GetChildF(pLunL0, "AttachedDriver");
2004 if (pLunAD)
2005 {
2006 CFGMR3RemoveNode(pLunAD);
2007 }
2008 else
2009 {
2010 CFGMR3RemoveNode(pLunL0);
2011 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
2012 rc = CFGMR3InsertString(pLunL0, "Driver", "NetSniffer"); RC_CHECK();
2013 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
2014 hrc = aNetworkAdapter->COMGETTER(TraceFile)(&str); H();
2015 if (str) /* check convention for indicating default file. */
2016 {
2017 rc = CFGMR3InsertStringW(pCfg, "File", str); RC_CHECK();
2018 }
2019 STR_FREE();
2020 }
2021 }
2022 else if (fAttachDetach && !fSniffer)
2023 {
2024 rc = PDMR3DeviceDetach(pVM, pszDevice, uInstance, uLun, 0 /*fFlags*/);
2025 if (rc == VINF_PDM_NO_DRIVER_ATTACHED_TO_LUN)
2026 rc = VINF_SUCCESS;
2027 AssertLogRelRCReturn(rc, rc);
2028
2029 /* nuke anything which might have been left behind. */
2030 CFGMR3RemoveNode(CFGMR3GetChildF(pInst, "LUN#%u", uLun));
2031 }
2032 else if (!fAttachDetach && fSniffer)
2033 {
2034 /* insert the sniffer filter driver. */
2035 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
2036 rc = CFGMR3InsertString(pLunL0, "Driver", "NetSniffer"); RC_CHECK();
2037 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
2038 hrc = aNetworkAdapter->COMGETTER(TraceFile)(&str); H();
2039 if (str) /* check convention for indicating default file. */
2040 {
2041 rc = CFGMR3InsertStringW(pCfg, "File", str); RC_CHECK();
2042 }
2043 STR_FREE();
2044 }
2045
2046 Bstr networkName, trunkName, trunkType;
2047 NetworkAttachmentType_T eAttachmentType;
2048 hrc = aNetworkAdapter->COMGETTER(AttachmentType)(&eAttachmentType); H();
2049 switch (eAttachmentType)
2050 {
2051 case NetworkAttachmentType_Null:
2052 break;
2053
2054 case NetworkAttachmentType_NAT:
2055 {
2056 if (fSniffer)
2057 {
2058 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL0); RC_CHECK();
2059 }
2060 else
2061 {
2062 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
2063 }
2064 rc = CFGMR3InsertString(pLunL0, "Driver", "NAT"); RC_CHECK();
2065 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
2066
2067 /* Configure TFTP prefix and boot filename. */
2068 hrc = virtualBox->COMGETTER(HomeFolder)(&str); H();
2069 if (str && *str)
2070 {
2071 rc = CFGMR3InsertStringF(pCfg, "TFTPPrefix", "%ls%c%s", str, RTPATH_DELIMITER, "TFTP"); RC_CHECK();
2072 }
2073 STR_FREE();
2074 hrc = pMachine->COMGETTER(Name)(&str); H();
2075 rc = CFGMR3InsertStringF(pCfg, "BootFile", "%ls.pxe", str); RC_CHECK();
2076 STR_FREE();
2077
2078 hrc = aNetworkAdapter->COMGETTER(NATNetwork)(&str); H();
2079 if (str && *str)
2080 {
2081 rc = CFGMR3InsertStringW(pCfg, "Network", str); RC_CHECK();
2082 /* NAT uses its own DHCP implementation */
2083 //networkName = Bstr(psz);
2084 }
2085 STR_FREE();
2086 break;
2087 }
2088
2089 case NetworkAttachmentType_Bridged:
2090 {
2091#if (defined(RT_OS_LINUX) || defined(RT_OS_FREEBSD)) && !defined(VBOX_WITH_NETFLT)
2092 hrc = pThis->attachToTapInterface(aNetworkAdapter);
2093 if (FAILED(hrc))
2094 {
2095 switch (hrc)
2096 {
2097 case VERR_ACCESS_DENIED:
2098 return VMSetError(pVM, VERR_HOSTIF_INIT_FAILED, RT_SRC_POS, N_(
2099 "Failed to open '/dev/net/tun' for read/write access. Please check the "
2100 "permissions of that node. Either run 'chmod 0666 /dev/net/tun' or "
2101 "change the group of that node and make yourself a member of that group. Make "
2102 "sure that these changes are permanent, especially if you are "
2103 "using udev"));
2104 default:
2105 AssertMsgFailed(("Could not attach to host interface! Bad!\n"));
2106 return VMSetError(pVM, VERR_HOSTIF_INIT_FAILED, RT_SRC_POS, N_(
2107 "Failed to initialize Host Interface Networking"));
2108 }
2109 }
2110
2111 Assert ((int)pThis->maTapFD[uInstance] >= 0);
2112 if ((int)pThis->maTapFD[uInstance] >= 0)
2113 {
2114 if (fSniffer)
2115 {
2116 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL0); RC_CHECK();
2117 }
2118 else
2119 {
2120 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
2121 }
2122 rc = CFGMR3InsertString(pLunL0, "Driver", "HostInterface"); RC_CHECK();
2123 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
2124 rc = CFGMR3InsertInteger(pCfg, "FileHandle", pThis->maTapFD[uInstance]); RC_CHECK();
2125 }
2126
2127#elif defined(VBOX_WITH_NETFLT)
2128 /*
2129 * This is the new VBoxNetFlt+IntNet stuff.
2130 */
2131 if (fSniffer)
2132 {
2133 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL0); RC_CHECK();
2134 }
2135 else
2136 {
2137 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0); RC_CHECK();
2138 }
2139
2140 Bstr HifName;
2141 hrc = aNetworkAdapter->COMGETTER(HostInterface)(HifName.asOutParam());
2142 if (FAILED(hrc))
2143 {
2144 LogRel(("NetworkAttachmentType_Bridged: COMGETTER(HostInterface) failed, hrc (0x%x)", hrc));
2145 H();
2146 }
2147
2148 Utf8Str HifNameUtf8(HifName);
2149 const char *pszHifName = HifNameUtf8.raw();
2150
2151# if defined(RT_OS_DARWIN)
2152 /* The name is on the form 'ifX: long name', chop it off at the colon. */
2153 char szTrunk[8];
2154 strncpy(szTrunk, pszHifName, sizeof(szTrunk));
2155 char *pszColon = (char *)memchr(szTrunk, ':', sizeof(szTrunk));
2156 if (!pszColon)
2157 {
2158 hrc = aNetworkAdapter->Detach(); H();
2159 return VMSetError(pVM, VERR_INTERNAL_ERROR, RT_SRC_POS,
2160 N_("Malformed host interface networking name '%ls'"),
2161 HifName.raw());
2162 }
2163 *pszColon = '\0';
2164 const char *pszTrunk = szTrunk;
2165
2166# elif defined(RT_OS_SOLARIS)
2167 /* The name is on the form format 'ifX[:1] - long name, chop it off at space. */
2168 char szTrunk[256];
2169 strlcpy(szTrunk, pszHifName, sizeof(szTrunk));
2170 char *pszSpace = (char *)memchr(szTrunk, ' ', sizeof(szTrunk));
2171
2172 /*
2173 * Currently don't bother about malformed names here for the sake of people using
2174 * VBoxManage and setting only the NIC name from there. If there is a space we
2175 * chop it off and proceed, otherwise just use whatever we've got.
2176 */
2177 if (pszSpace)
2178 *pszSpace = '\0';
2179
2180 /* Chop it off at the colon (zone naming eg: e1000g:1 we need only the e1000g) */
2181 char *pszColon = (char *)memchr(szTrunk, ':', sizeof(szTrunk));
2182 if (pszColon)
2183 *pszColon = '\0';
2184
2185 const char *pszTrunk = szTrunk;
2186
2187# elif defined(RT_OS_WINDOWS)
2188 ComPtr<IHostNetworkInterface> hostInterface;
2189 hrc = host->FindHostNetworkInterfaceByName(HifName, hostInterface.asOutParam());
2190 if (!SUCCEEDED(hrc))
2191 {
2192 AssertLogRelMsgFailed(("NetworkAttachmentType_Bridged: FindByName failed, rc=%Rhrc (0x%x)", hrc, hrc));
2193 return VMSetError(pVM, VERR_INTERNAL_ERROR, RT_SRC_POS,
2194 N_("Inexistent host networking interface, name '%ls'"),
2195 HifName.raw());
2196 }
2197
2198 HostNetworkInterfaceType_T eIfType;
2199 hrc = hostInterface->COMGETTER(InterfaceType)(&eIfType);
2200 if (FAILED(hrc))
2201 {
2202 LogRel(("NetworkAttachmentType_Bridged: COMGETTER(InterfaceType) failed, hrc (0x%x)", hrc));
2203 H();
2204 }
2205
2206 if (eIfType != HostNetworkInterfaceType_Bridged)
2207 {
2208 return VMSetError(pVM, VERR_INTERNAL_ERROR, RT_SRC_POS,
2209 N_("Interface ('%ls') is not a Bridged Adapter interface"),
2210 HifName.raw());
2211 }
2212
2213 hrc = hostInterface->COMGETTER(Id)(&str);
2214 if (FAILED(hrc))
2215 {
2216 LogRel(("NetworkAttachmentType_Bridged: COMGETTER(Id) failed, hrc (0x%x)", hrc));
2217 H();
2218 }
2219 Guid hostIFGuid(str);
2220 STR_FREE();
2221
2222 INetCfg *pNc;
2223 ComPtr<INetCfgComponent> pAdaptorComponent;
2224 LPWSTR pszApp;
2225 int rc = VERR_INTNET_FLT_IF_NOT_FOUND;
2226
2227 hrc = VBoxNetCfgWinQueryINetCfg(FALSE /*fGetWriteLock*/,
2228 L"VirtualBox",
2229 &pNc,
2230 &pszApp);
2231 Assert(hrc == S_OK);
2232 if (hrc == S_OK)
2233 {
2234 /* get the adapter's INetCfgComponent*/
2235 hrc = VBoxNetCfgWinGetComponentByGuid(pNc, &GUID_DEVCLASS_NET, (GUID*)hostIFGuid.ptr(), pAdaptorComponent.asOutParam());
2236 if (hrc != S_OK)
2237 {
2238 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
2239 LogRel(("NetworkAttachmentType_Bridged: VBoxNetCfgWinGetComponentByGuid failed, hrc (0x%x)", hrc));
2240 H();
2241 }
2242 }
2243#define VBOX_WIN_BINDNAME_PREFIX "\\DEVICE\\"
2244 char szTrunkName[INTNET_MAX_TRUNK_NAME];
2245 char *pszTrunkName = szTrunkName;
2246 wchar_t * pswzBindName;
2247 hrc = pAdaptorComponent->GetBindName(&pswzBindName);
2248 Assert(hrc == S_OK);
2249 if (hrc == S_OK)
2250 {
2251 int cwBindName = (int)wcslen(pswzBindName) + 1;
2252 int cbFullBindNamePrefix = sizeof(VBOX_WIN_BINDNAME_PREFIX);
2253 if (sizeof(szTrunkName) > cbFullBindNamePrefix + cwBindName)
2254 {
2255 strcpy(szTrunkName, VBOX_WIN_BINDNAME_PREFIX);
2256 pszTrunkName += cbFullBindNamePrefix-1;
2257 if (!WideCharToMultiByte(CP_ACP, 0, pswzBindName, cwBindName, pszTrunkName,
2258 sizeof(szTrunkName) - cbFullBindNamePrefix + 1, NULL, NULL))
2259 {
2260 DWORD err = GetLastError();
2261 hrc = HRESULT_FROM_WIN32(err);
2262 AssertMsgFailed(("%hrc=%Rhrc %#x\n", hrc, hrc));
2263 AssertLogRelMsgFailed(("NetworkAttachmentType_Bridged: WideCharToMultiByte failed, hr=%Rhrc (0x%x) err=%u\n", hrc, hrc, err));
2264 }
2265 }
2266 else
2267 {
2268 AssertLogRelMsgFailed(("NetworkAttachmentType_Bridged: insufficient szTrunkName buffer space\n"));
2269 /** @todo set appropriate error code */
2270 hrc = E_FAIL;
2271 }
2272
2273 if (hrc != S_OK)
2274 {
2275 AssertFailed();
2276 CoTaskMemFree(pswzBindName);
2277 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
2278 H();
2279 }
2280
2281 /* we're not freeing the bind name since we'll use it later for detecting wireless*/
2282 }
2283 else
2284 {
2285 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
2286 AssertLogRelMsgFailed(("NetworkAttachmentType_Bridged: VBoxNetCfgWinGetComponentByGuid failed, hrc (0x%x)", hrc));
2287 H();
2288 }
2289 const char *pszTrunk = szTrunkName;
2290 /* we're not releasing the INetCfg stuff here since we use it later to figure out whether it is wireless */
2291
2292# elif defined(RT_OS_LINUX) || defined(RT_OS_FREEBSD)
2293 /** @todo Check for malformed names. */
2294 const char *pszTrunk = pszHifName;
2295
2296# else
2297# error "PORTME (VBOX_WITH_NETFLT)"
2298# endif
2299
2300 rc = CFGMR3InsertString(pLunL0, "Driver", "IntNet"); RC_CHECK();
2301 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
2302 rc = CFGMR3InsertString(pCfg, "Trunk", pszTrunk); RC_CHECK();
2303 rc = CFGMR3InsertInteger(pCfg, "TrunkType", kIntNetTrunkType_NetFlt);
2304 RC_CHECK();
2305 char szNetwork[INTNET_MAX_NETWORK_NAME];
2306 RTStrPrintf(szNetwork, sizeof(szNetwork), "HostInterfaceNetworking-%s", pszHifName);
2307 rc = CFGMR3InsertString(pCfg, "Network", szNetwork); RC_CHECK();
2308 networkName = Bstr(szNetwork);
2309 trunkName = Bstr(pszTrunk);
2310 trunkType = Bstr(TRUNKTYPE_NETFLT);
2311
2312# if defined(RT_OS_DARWIN)
2313 /** @todo Come up with a better deal here. Problem is that IHostNetworkInterface is completely useless here. */
2314 if ( strstr(pszHifName, "Wireless")
2315 || strstr(pszHifName, "AirPort" ))
2316 {
2317 rc = CFGMR3InsertInteger(pCfg, "SharedMacOnWire", true); RC_CHECK();
2318 }
2319# elif defined(RT_OS_LINUX)
2320 int iSock = socket(AF_INET, SOCK_DGRAM, 0);
2321 if (iSock >= 0)
2322 {
2323 struct iwreq WRq;
2324
2325 memset(&WRq, 0, sizeof(WRq));
2326 strncpy(WRq.ifr_name, pszHifName, IFNAMSIZ);
2327 bool fSharedMacOnWire = ioctl(iSock, SIOCGIWNAME, &WRq) >= 0;
2328 close(iSock);
2329 if (fSharedMacOnWire)
2330 {
2331 rc = CFGMR3InsertInteger(pCfg, "SharedMacOnWire", true);
2332 RC_CHECK();
2333 Log(("Set SharedMacOnWire\n"));
2334 }
2335 else
2336 Log(("Failed to get wireless name\n"));
2337 }
2338 else
2339 Log(("Failed to open wireless socket\n"));
2340# elif defined(RT_OS_WINDOWS)
2341# define DEVNAME_PREFIX L"\\\\.\\"
2342 /* we are getting the medium type via IOCTL_NDIS_QUERY_GLOBAL_STATS Io Control
2343 * there is a pretty long way till there though since we need to obtain the symbolic link name
2344 * for the adapter device we are going to query given the device Guid */
2345
2346
2347 /* prepend the "\\\\.\\" to the bind name to obtain the link name */
2348
2349 wchar_t FileName[MAX_PATH];
2350 wcscpy(FileName, DEVNAME_PREFIX);
2351 wcscpy((wchar_t*)(((char*)FileName) + sizeof(DEVNAME_PREFIX) - sizeof(FileName[0])), pswzBindName);
2352
2353 /* open the device */
2354 HANDLE hDevice = CreateFile(FileName,
2355 GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE,
2356 NULL,
2357 OPEN_EXISTING,
2358 FILE_ATTRIBUTE_NORMAL,
2359 NULL);
2360
2361 if (hDevice != INVALID_HANDLE_VALUE)
2362 {
2363 bool fSharedMacOnWire = false;
2364
2365 /* now issue the OID_GEN_PHYSICAL_MEDIUM query */
2366 DWORD Oid = OID_GEN_PHYSICAL_MEDIUM;
2367 NDIS_PHYSICAL_MEDIUM PhMedium;
2368 DWORD cbResult;
2369 if (DeviceIoControl(hDevice,
2370 IOCTL_NDIS_QUERY_GLOBAL_STATS,
2371 &Oid,
2372 sizeof(Oid),
2373 &PhMedium,
2374 sizeof(PhMedium),
2375 &cbResult,
2376 NULL))
2377 {
2378 /* that was simple, now examine PhMedium */
2379 if ( PhMedium == NdisPhysicalMediumWirelessWan
2380 || PhMedium == NdisPhysicalMediumWirelessLan
2381 || PhMedium == NdisPhysicalMediumNative802_11
2382 || PhMedium == NdisPhysicalMediumBluetooth)
2383 fSharedMacOnWire = true;
2384 }
2385 else
2386 {
2387 int winEr = GetLastError();
2388 LogRel(("Console::configConstructor: DeviceIoControl failed, err (0x%x), ignoring\n", winEr));
2389 Assert(winEr == ERROR_INVALID_PARAMETER || winEr == ERROR_NOT_SUPPORTED || winEr == ERROR_BAD_COMMAND);
2390 }
2391 CloseHandle(hDevice);
2392
2393 if (fSharedMacOnWire)
2394 {
2395 Log(("this is a wireless adapter"));
2396 rc = CFGMR3InsertInteger(pCfg, "SharedMacOnWire", true); RC_CHECK();
2397 Log(("Set SharedMacOnWire\n"));
2398 }
2399 else
2400 Log(("this is NOT a wireless adapter"));
2401 }
2402 else
2403 {
2404 int winEr = GetLastError();
2405 AssertLogRelMsgFailed(("Console::configConstructor: CreateFile failed, err (0x%x), ignoring\n", winEr));
2406 }
2407
2408 CoTaskMemFree(pswzBindName);
2409
2410 pAdaptorComponent.setNull();
2411 /* release the pNc finally */
2412 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
2413# else
2414 /** @todo PORTME: wireless detection */
2415# endif
2416
2417# if defined(RT_OS_SOLARIS)
2418# if 0 /* bird: this is a bit questionable and might cause more trouble than its worth. */
2419 /* Zone access restriction, don't allow snopping the global zone. */
2420 zoneid_t ZoneId = getzoneid();
2421 if (ZoneId != GLOBAL_ZONEID)
2422 {
2423 rc = CFGMR3InsertInteger(pCfg, "IgnoreAllPromisc", true); RC_CHECK();
2424 }
2425# endif
2426# endif
2427
2428#elif defined(RT_OS_WINDOWS) /* not defined NetFlt */
2429 /* NOTHING TO DO HERE */
2430#elif defined(RT_OS_LINUX)
2431/// @todo aleksey: is there anything to be done here?
2432#elif defined(RT_OS_FREEBSD)
2433/** @todo FreeBSD: Check out this later (HIF networking). */
2434#else
2435# error "Port me"
2436#endif
2437 break;
2438 }
2439
2440 case NetworkAttachmentType_Internal:
2441 {
2442 hrc = aNetworkAdapter->COMGETTER(InternalNetwork)(&str); H();
2443 if (str && *str)
2444 {
2445 if (fSniffer)
2446 {
2447 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL0);
2448 RC_CHECK();
2449 }
2450 else
2451 {
2452 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0);
2453 RC_CHECK();
2454 }
2455 rc = CFGMR3InsertString(pLunL0, "Driver", "IntNet"); RC_CHECK();
2456 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
2457 rc = CFGMR3InsertStringW(pCfg, "Network", str); RC_CHECK();
2458 rc = CFGMR3InsertInteger(pCfg, "TrunkType", kIntNetTrunkType_WhateverNone); RC_CHECK();
2459 networkName = str;
2460 trunkType = Bstr(TRUNKTYPE_WHATEVER);
2461 }
2462 STR_FREE();
2463 break;
2464 }
2465
2466 case NetworkAttachmentType_HostOnly:
2467 {
2468 if (fSniffer)
2469 {
2470 rc = CFGMR3InsertNode(pLunL0, "AttachedDriver", &pLunL0);
2471 RC_CHECK();
2472 }
2473 else
2474 {
2475 rc = CFGMR3InsertNode(pInst, "LUN#0", &pLunL0);
2476 RC_CHECK();
2477 }
2478
2479 rc = CFGMR3InsertString(pLunL0, "Driver", "IntNet"); RC_CHECK();
2480 rc = CFGMR3InsertNode(pLunL0, "Config", &pCfg); RC_CHECK();
2481
2482 Bstr HifName;
2483 hrc = aNetworkAdapter->COMGETTER(HostInterface)(HifName.asOutParam());
2484 if (FAILED(hrc))
2485 {
2486 LogRel(("NetworkAttachmentType_HostOnly: COMGETTER(HostInterface) failed, hrc (0x%x)\n", hrc));
2487 H();
2488 }
2489
2490 Utf8Str HifNameUtf8(HifName);
2491 const char *pszHifName = HifNameUtf8.raw();
2492 LogRel(("NetworkAttachmentType_HostOnly: COMGETTER(HostInterface): %s\n", pszHifName));
2493 ComPtr<IHostNetworkInterface> hostInterface;
2494 rc = host->FindHostNetworkInterfaceByName(HifName, hostInterface.asOutParam());
2495 if (!SUCCEEDED(rc))
2496 {
2497 LogRel(("NetworkAttachmentType_HostOnly: FindByName failed, rc (0x%x)\n", rc));
2498 return VMSetError(pVM, VERR_INTERNAL_ERROR, RT_SRC_POS,
2499 N_("Inexistent host networking interface, name '%ls'"),
2500 HifName.raw());
2501 }
2502
2503 char szNetwork[INTNET_MAX_NETWORK_NAME];
2504 RTStrPrintf(szNetwork, sizeof(szNetwork), "HostInterfaceNetworking-%s", pszHifName);
2505
2506#if defined(RT_OS_WINDOWS)
2507# ifndef VBOX_WITH_NETFLT
2508 hrc = E_NOTIMPL;
2509 LogRel(("NetworkAttachmentType_HostOnly: Not Implemented\n"));
2510 H();
2511# else /* defined VBOX_WITH_NETFLT*/
2512 /** @todo r=bird: Put this in a function. */
2513
2514 HostNetworkInterfaceType_T eIfType;
2515 hrc = hostInterface->COMGETTER(InterfaceType)(&eIfType);
2516 if (FAILED(hrc))
2517 {
2518 LogRel(("NetworkAttachmentType_HostOnly: COMGETTER(InterfaceType) failed, hrc (0x%x)\n", hrc));
2519 H();
2520 }
2521
2522 if (eIfType != HostNetworkInterfaceType_HostOnly)
2523 return VMSetError(pVM, VERR_INTERNAL_ERROR, RT_SRC_POS,
2524 N_("Interface ('%ls') is not a Host-Only Adapter interface"),
2525 HifName.raw());
2526
2527 hrc = hostInterface->COMGETTER(Id)(&str);
2528 if (FAILED(hrc))
2529 {
2530 LogRel(("NetworkAttachmentType_HostOnly: COMGETTER(Id) failed, hrc (0x%x)\n", hrc));
2531 H();
2532 }
2533 Guid hostIFGuid(str);
2534 STR_FREE();
2535
2536 INetCfg *pNc;
2537 ComPtr<INetCfgComponent> pAdaptorComponent;
2538 LPWSTR pszApp;
2539 rc = VERR_INTNET_FLT_IF_NOT_FOUND;
2540
2541 hrc = VBoxNetCfgWinQueryINetCfg(FALSE,
2542 L"VirtualBox",
2543 &pNc,
2544 &pszApp);
2545 Assert(hrc == S_OK);
2546 if (hrc == S_OK)
2547 {
2548 /* get the adapter's INetCfgComponent*/
2549 hrc = VBoxNetCfgWinGetComponentByGuid(pNc, &GUID_DEVCLASS_NET, (GUID*)hostIFGuid.ptr(), pAdaptorComponent.asOutParam());
2550 if (hrc != S_OK)
2551 {
2552 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
2553 LogRel(("NetworkAttachmentType_HostOnly: VBoxNetCfgWinGetComponentByGuid failed, hrc=%Rhrc (0x%x)\n", hrc, hrc));
2554 H();
2555 }
2556 }
2557#define VBOX_WIN_BINDNAME_PREFIX "\\DEVICE\\"
2558 char szTrunkName[INTNET_MAX_TRUNK_NAME];
2559 char *pszTrunkName = szTrunkName;
2560 wchar_t * pswzBindName;
2561 hrc = pAdaptorComponent->GetBindName(&pswzBindName);
2562 Assert(hrc == S_OK);
2563 if (hrc == S_OK)
2564 {
2565 int cwBindName = (int)wcslen(pswzBindName) + 1;
2566 int cbFullBindNamePrefix = sizeof(VBOX_WIN_BINDNAME_PREFIX);
2567 if (sizeof(szTrunkName) > cbFullBindNamePrefix + cwBindName)
2568 {
2569 strcpy(szTrunkName, VBOX_WIN_BINDNAME_PREFIX);
2570 pszTrunkName += cbFullBindNamePrefix-1;
2571 if (!WideCharToMultiByte(CP_ACP, 0, pswzBindName, cwBindName, pszTrunkName,
2572 sizeof(szTrunkName) - cbFullBindNamePrefix + 1, NULL, NULL))
2573 {
2574 DWORD err = GetLastError();
2575 hrc = HRESULT_FROM_WIN32(err);
2576 AssertLogRelMsgFailed(("NetworkAttachmentType_HostOnly: WideCharToMultiByte failed, hr=%Rhrc (0x%x) err=%u\n", hrc, hrc, err));
2577 }
2578 }
2579 else
2580 {
2581 AssertLogRelMsgFailed(("NetworkAttachmentType_HostOnly: insufficient szTrunkName buffer space\n"));
2582 /** @todo set appropriate error code */
2583 hrc = E_FAIL;
2584 }
2585
2586 if (hrc != S_OK)
2587 {
2588 AssertFailed();
2589 CoTaskMemFree(pswzBindName);
2590 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
2591 H();
2592 }
2593 }
2594 else
2595 {
2596 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
2597 AssertLogRelMsgFailed(("NetworkAttachmentType_HostOnly: VBoxNetCfgWinGetComponentByGuid failed, hrc=%Rhrc (0x%x)\n", hrc, hrc));
2598 H();
2599 }
2600
2601
2602 CoTaskMemFree(pswzBindName);
2603
2604 pAdaptorComponent.setNull();
2605 /* release the pNc finally */
2606 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE /*fHasWriteLock*/);
2607
2608 const char *pszTrunk = szTrunkName;
2609
2610 rc = CFGMR3InsertInteger(pCfg, "TrunkType", kIntNetTrunkType_NetAdp); RC_CHECK();
2611 rc = CFGMR3InsertString(pCfg, "Trunk", pszTrunk); RC_CHECK();
2612 rc = CFGMR3InsertString(pCfg, "Network", szNetwork); RC_CHECK();
2613 networkName = Bstr(szNetwork);
2614 trunkName = Bstr(pszTrunk);
2615 trunkType = TRUNKTYPE_NETADP;
2616# endif /* defined VBOX_WITH_NETFLT*/
2617#elif defined(RT_OS_DARWIN)
2618 rc = CFGMR3InsertString(pCfg, "Trunk", pszHifName); RC_CHECK();
2619 rc = CFGMR3InsertString(pCfg, "Network", szNetwork); RC_CHECK();
2620 rc = CFGMR3InsertInteger(pCfg, "TrunkType", kIntNetTrunkType_NetAdp); RC_CHECK();
2621 networkName = Bstr(szNetwork);
2622 trunkName = Bstr(pszHifName);
2623 trunkType = TRUNKTYPE_NETADP;
2624#else
2625 rc = CFGMR3InsertString(pCfg, "Trunk", pszHifName); RC_CHECK();
2626 rc = CFGMR3InsertString(pCfg, "Network", szNetwork); RC_CHECK();
2627 rc = CFGMR3InsertInteger(pCfg, "TrunkType", kIntNetTrunkType_NetFlt); RC_CHECK();
2628 networkName = Bstr(szNetwork);
2629 trunkName = Bstr(pszHifName);
2630 trunkType = TRUNKTYPE_NETFLT;
2631#endif
2632#if !defined(RT_OS_WINDOWS) && defined(VBOX_WITH_NETFLT)
2633
2634 Bstr tmpAddr, tmpMask;
2635
2636 hrc = virtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPAddress", pszHifName), tmpAddr.asOutParam());
2637 if (SUCCEEDED(hrc) && !tmpAddr.isEmpty())
2638 {
2639 hrc = virtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPNetMask", pszHifName), tmpMask.asOutParam());
2640 if (SUCCEEDED(hrc) && !tmpMask.isEmpty())
2641 hrc = hostInterface->EnableStaticIpConfig(tmpAddr, tmpMask);
2642 else
2643 hrc = hostInterface->EnableStaticIpConfig(tmpAddr,
2644 Bstr(VBOXNET_IPV4MASK_DEFAULT));
2645 }
2646 else
2647 hrc = hostInterface->EnableStaticIpConfig(Bstr(VBOXNET_IPV4ADDR_DEFAULT),
2648 Bstr(VBOXNET_IPV4MASK_DEFAULT));
2649 ComAssertComRC(hrc); /** @todo r=bird: Why this isn't fatal? (H()) */
2650
2651 hrc = virtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPV6Address", pszHifName), tmpAddr.asOutParam());
2652 if (SUCCEEDED(hrc))
2653 hrc = virtualBox->GetExtraData(BstrFmt("HostOnly/%s/IPV6NetMask", pszHifName), tmpMask.asOutParam());
2654 if (SUCCEEDED(hrc) && !tmpAddr.isEmpty() && !tmpMask.isEmpty())
2655 {
2656 hrc = hostInterface->EnableStaticIpConfigV6(tmpAddr, Utf8Str(tmpMask).toUInt32());
2657 ComAssertComRC(hrc); /** @todo r=bird: Why this isn't fatal? (H()) */
2658 }
2659#endif
2660 break;
2661 }
2662
2663 default:
2664 AssertMsgFailed(("should not get here!\n"));
2665 break;
2666 }
2667
2668 /*
2669 * Attempt to attach the driver.
2670 */
2671 switch (eAttachmentType)
2672 {
2673 case NetworkAttachmentType_Null:
2674 break;
2675
2676 case NetworkAttachmentType_Bridged:
2677 case NetworkAttachmentType_Internal:
2678 case NetworkAttachmentType_HostOnly:
2679 case NetworkAttachmentType_NAT:
2680 {
2681 if (SUCCEEDED(hrc) && SUCCEEDED(rc))
2682 {
2683 if (fAttachDetach)
2684 {
2685 rc = PDMR3DriverAttach(pVM, pszDevice, uInstance, uLun, 0 /*fFlags*/, NULL /* ppBase */);
2686 AssertRC(rc);
2687 }
2688
2689 {
2690 /** @todo pritesh: get the dhcp server name from the
2691 * previous network configuration and then stop the server
2692 * else it may conflict with the dhcp server running with
2693 * the current attachment type
2694 */
2695 /* Stop the hostonly DHCP Server */
2696 }
2697
2698 if (!networkName.isNull())
2699 {
2700 /*
2701 * Until we implement service reference counters DHCP Server will be stopped
2702 * by DHCPServerRunner destructor.
2703 */
2704 ComPtr<IDHCPServer> dhcpServer;
2705 hrc = virtualBox->FindDHCPServerByNetworkName(networkName.mutableRaw(), dhcpServer.asOutParam());
2706 if (SUCCEEDED(hrc))
2707 {
2708 /* there is a DHCP server available for this network */
2709 BOOL fEnabled;
2710 hrc = dhcpServer->COMGETTER(Enabled)(&fEnabled);
2711 if (FAILED(hrc))
2712 {
2713 LogRel(("DHCP svr: COMGETTER(Enabled) failed, hrc (%Rhrc)", hrc));
2714 H();
2715 }
2716
2717 if (fEnabled)
2718 hrc = dhcpServer->Start(networkName, trunkName, trunkType);
2719 }
2720 else
2721 hrc = S_OK;
2722 }
2723 }
2724
2725 break;
2726 }
2727
2728 default:
2729 AssertMsgFailed(("should not get here!\n"));
2730 break;
2731 }
2732
2733 pThis->meAttachmentType[uInstance] = eAttachmentType;
2734
2735#undef STR_FREE
2736#undef H
2737#undef RC_CHECK
2738
2739 return VINF_SUCCESS;
2740}
2741
2742#ifdef VBOX_WITH_GUEST_PROPS
2743/**
2744 * Set an array of guest properties
2745 */
2746static void configSetProperties(VMMDev * const pVMMDev, void *names,
2747 void *values, void *timestamps, void *flags)
2748{
2749 VBOXHGCMSVCPARM parms[4];
2750
2751 parms[0].type = VBOX_HGCM_SVC_PARM_PTR;
2752 parms[0].u.pointer.addr = names;
2753 parms[0].u.pointer.size = 0; /* We don't actually care. */
2754 parms[1].type = VBOX_HGCM_SVC_PARM_PTR;
2755 parms[1].u.pointer.addr = values;
2756 parms[1].u.pointer.size = 0; /* We don't actually care. */
2757 parms[2].type = VBOX_HGCM_SVC_PARM_PTR;
2758 parms[2].u.pointer.addr = timestamps;
2759 parms[2].u.pointer.size = 0; /* We don't actually care. */
2760 parms[3].type = VBOX_HGCM_SVC_PARM_PTR;
2761 parms[3].u.pointer.addr = flags;
2762 parms[3].u.pointer.size = 0; /* We don't actually care. */
2763
2764 pVMMDev->hgcmHostCall ("VBoxGuestPropSvc", guestProp::SET_PROPS_HOST, 4,
2765 &parms[0]);
2766}
2767
2768/**
2769 * Set a single guest property
2770 */
2771static void configSetProperty(VMMDev * const pVMMDev, const char *pszName,
2772 const char *pszValue, const char *pszFlags)
2773{
2774 VBOXHGCMSVCPARM parms[4];
2775
2776 AssertPtrReturnVoid(pszName);
2777 AssertPtrReturnVoid(pszValue);
2778 AssertPtrReturnVoid(pszFlags);
2779 parms[0].type = VBOX_HGCM_SVC_PARM_PTR;
2780 parms[0].u.pointer.addr = (void *)pszName;
2781 parms[0].u.pointer.size = strlen(pszName) + 1;
2782 parms[1].type = VBOX_HGCM_SVC_PARM_PTR;
2783 parms[1].u.pointer.addr = (void *)pszValue;
2784 parms[1].u.pointer.size = strlen(pszValue) + 1;
2785 parms[2].type = VBOX_HGCM_SVC_PARM_PTR;
2786 parms[2].u.pointer.addr = (void *)pszFlags;
2787 parms[2].u.pointer.size = strlen(pszFlags) + 1;
2788 pVMMDev->hgcmHostCall ("VBoxGuestPropSvc", guestProp::SET_PROP_HOST, 3,
2789 &parms[0]);
2790}
2791#endif /* VBOX_WITH_GUEST_PROPS */
2792
2793/**
2794 * Set up the Guest Property service, populate it with properties read from
2795 * the machine XML and set a couple of initial properties.
2796 */
2797/* static */ int Console::configGuestProperties(void *pvConsole)
2798{
2799#ifdef VBOX_WITH_GUEST_PROPS
2800 AssertReturn(pvConsole, VERR_GENERAL_FAILURE);
2801 ComObjPtr<Console> pConsole = static_cast <Console *> (pvConsole);
2802
2803 /* Load the service */
2804 int rc = pConsole->mVMMDev->hgcmLoadService ("VBoxGuestPropSvc", "VBoxGuestPropSvc");
2805
2806 if (RT_FAILURE(rc))
2807 {
2808 LogRel(("VBoxGuestPropSvc is not available. rc = %Rrc\n", rc));
2809 /* That is not a fatal failure. */
2810 rc = VINF_SUCCESS;
2811 }
2812 else
2813 {
2814 /*
2815 * Initialize built-in properties that can be changed and saved.
2816 *
2817 * These are typically transient properties that the guest cannot
2818 * change.
2819 */
2820
2821 /* Sysprep execution by VBoxService. */
2822 configSetProperty(pConsole->mVMMDev,
2823 "/VirtualBox/HostGuest/SysprepExec", "",
2824 "TRANSIENT, RDONLYGUEST");
2825 configSetProperty(pConsole->mVMMDev,
2826 "/VirtualBox/HostGuest/SysprepArgs", "",
2827 "TRANSIENT, RDONLYGUEST");
2828
2829 /*
2830 * Pull over the properties from the server.
2831 */
2832 SafeArray<BSTR> namesOut;
2833 SafeArray<BSTR> valuesOut;
2834 SafeArray<ULONG64> timestampsOut;
2835 SafeArray<BSTR> flagsOut;
2836 HRESULT hrc = pConsole->mControl->PullGuestProperties
2837 (ComSafeArrayAsOutParam(namesOut),
2838 ComSafeArrayAsOutParam(valuesOut),
2839 ComSafeArrayAsOutParam(timestampsOut),
2840 ComSafeArrayAsOutParam(flagsOut));
2841 AssertMsgReturn(SUCCEEDED(hrc), ("hrc=%#x\n", hrc),
2842 VERR_GENERAL_FAILURE);
2843 size_t cProps = namesOut.size();
2844 size_t cAlloc = cProps + 1;
2845 if ( valuesOut.size() != cProps
2846 || timestampsOut.size() != cProps
2847 || flagsOut.size() != cProps
2848 )
2849 AssertFailedReturn(VERR_INVALID_PARAMETER);
2850
2851 char **papszNames, **papszValues, **papszFlags;
2852 char szEmpty[] = "";
2853 ULONG64 *pau64Timestamps;
2854 papszNames = (char **)RTMemTmpAllocZ(sizeof(void *) * cAlloc);
2855 papszValues = (char **)RTMemTmpAllocZ(sizeof(void *) * cAlloc);
2856 pau64Timestamps = (ULONG64 *)RTMemTmpAllocZ(sizeof(ULONG64) * cAlloc);
2857 papszFlags = (char **)RTMemTmpAllocZ(sizeof(void *) * cAlloc);
2858 if (papszNames && papszValues && pau64Timestamps && papszFlags)
2859 {
2860 for (unsigned i = 0; RT_SUCCESS(rc) && i < cProps; ++i)
2861 {
2862 AssertPtrReturn(namesOut[i], VERR_INVALID_PARAMETER);
2863 rc = RTUtf16ToUtf8(namesOut[i], &papszNames[i]);
2864 if (RT_FAILURE(rc))
2865 break;
2866 if (valuesOut[i])
2867 rc = RTUtf16ToUtf8(valuesOut[i], &papszValues[i]);
2868 else
2869 papszValues[i] = szEmpty;
2870 if (RT_FAILURE(rc))
2871 break;
2872 pau64Timestamps[i] = timestampsOut[i];
2873 if (flagsOut[i])
2874 rc = RTUtf16ToUtf8(flagsOut[i], &papszFlags[i]);
2875 else
2876 papszFlags[i] = szEmpty;
2877 }
2878 if (RT_SUCCESS(rc))
2879 configSetProperties(pConsole->mVMMDev,
2880 (void *)papszNames,
2881 (void *)papszValues,
2882 (void *)pau64Timestamps,
2883 (void *)papszFlags);
2884 for (unsigned i = 0; i < cProps; ++i)
2885 {
2886 RTStrFree(papszNames[i]);
2887 if (valuesOut[i])
2888 RTStrFree(papszValues[i]);
2889 if (flagsOut[i])
2890 RTStrFree(papszFlags[i]);
2891 }
2892 }
2893 else
2894 rc = VERR_NO_MEMORY;
2895 RTMemTmpFree(papszNames);
2896 RTMemTmpFree(papszValues);
2897 RTMemTmpFree(pau64Timestamps);
2898 RTMemTmpFree(papszFlags);
2899 AssertRCReturn(rc, rc);
2900
2901 /*
2902 * These properties have to be set before pulling over the properties
2903 * from the machine XML, to ensure that properties saved in the XML
2904 * will override them.
2905 */
2906 /* Set the VBox version string as a guest property */
2907 configSetProperty(pConsole->mVMMDev, "/VirtualBox/HostInfo/VBoxVer",
2908 VBOX_VERSION_STRING, "TRANSIENT, RDONLYGUEST");
2909 /* Set the VBox SVN revision as a guest property */
2910 configSetProperty(pConsole->mVMMDev, "/VirtualBox/HostInfo/VBoxRev",
2911 RTBldCfgRevisionStr(), "TRANSIENT, RDONLYGUEST");
2912
2913 /*
2914 * Register the host notification callback
2915 */
2916 HGCMSVCEXTHANDLE hDummy;
2917 HGCMHostRegisterServiceExtension(&hDummy, "VBoxGuestPropSvc",
2918 Console::doGuestPropNotification,
2919 pvConsole);
2920
2921 Log(("Set VBoxGuestPropSvc property store\n"));
2922 }
2923 return VINF_SUCCESS;
2924#else /* !VBOX_WITH_GUEST_PROPS */
2925 return VERR_NOT_SUPPORTED;
2926#endif /* !VBOX_WITH_GUEST_PROPS */
2927}
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