VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/HostImpl.cpp@ 59381

Last change on this file since 59381 was 59117, checked in by vboxsync, 9 years ago

USB,Main: Rework USBProxyService. Split it into a USBProxyService and USBProxyBackend class, USBProxyService can use multiple USBProxyBackend instances as sources for USB devices to attach to a VM which will be used for USB/IP support. Change the PDM USB API to contain a backend parameter instead of a remote flag to indicate the USB backend to use for the given device.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 115.7 KB
Line 
1/* $Id: HostImpl.cpp 59117 2015-12-14 14:04:37Z vboxsync $ */
2/** @file
3 * VirtualBox COM class implementation: Host
4 */
5
6/*
7 * Copyright (C) 2004-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18#define __STDC_LIMIT_MACROS
19#define __STDC_CONSTANT_MACROS
20
21// for some reason Windows burns in sdk\...\winsock.h if this isn't included first
22#include "VBox/com/ptr.h"
23
24#include "HostImpl.h"
25
26#ifdef VBOX_WITH_USB
27# include "HostUSBDeviceImpl.h"
28# include "USBDeviceFilterImpl.h"
29# include "USBProxyService.h"
30#else
31# include "VirtualBoxImpl.h"
32#endif // VBOX_WITH_USB
33
34#include "HostNetworkInterfaceImpl.h"
35#include "HostVideoInputDeviceImpl.h"
36#include "MachineImpl.h"
37#include "AutoCaller.h"
38#include "Logging.h"
39#include "Performance.h"
40
41#include "MediumImpl.h"
42#include "HostPower.h"
43
44#if defined(RT_OS_LINUX) || defined(RT_OS_FREEBSD)
45# include <HostHardwareLinux.h>
46#endif
47
48#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
49# include <set>
50#endif
51
52#ifdef VBOX_WITH_RESOURCE_USAGE_API
53# include "PerformanceImpl.h"
54#endif /* VBOX_WITH_RESOURCE_USAGE_API */
55
56#if defined(RT_OS_WINDOWS) && defined(VBOX_WITH_NETFLT)
57# include <VBox/VBoxNetCfg-win.h>
58#endif /* #if defined(RT_OS_WINDOWS) && defined(VBOX_WITH_NETFLT) */
59
60#if defined(RT_OS_DARWIN) && ARCH_BITS == 32
61# include <sys/types.h>
62# include <sys/sysctl.h>
63#endif
64
65#ifdef RT_OS_LINUX
66# include <sys/ioctl.h>
67# include <errno.h>
68# include <net/if.h>
69# include <net/if_arp.h>
70#endif /* RT_OS_LINUX */
71
72#ifdef RT_OS_SOLARIS
73# include <fcntl.h>
74# include <unistd.h>
75# include <stropts.h>
76# include <errno.h>
77# include <limits.h>
78# include <stdio.h>
79# include <libdevinfo.h>
80# include <sys/mkdev.h>
81# include <sys/scsi/generic/inquiry.h>
82# include <net/if.h>
83# include <sys/socket.h>
84# include <sys/sockio.h>
85# include <net/if_arp.h>
86# include <net/if.h>
87# include <sys/types.h>
88# include <sys/stat.h>
89# include <sys/cdio.h>
90# include <sys/dkio.h>
91# include <sys/mnttab.h>
92# include <sys/mntent.h>
93/* Dynamic loading of libhal on Solaris hosts */
94# ifdef VBOX_USE_LIBHAL
95# include "vbox-libhal.h"
96extern "C" char *getfullrawname(char *);
97# endif
98# include "solaris/DynLoadLibSolaris.h"
99
100/**
101 * Solaris DVD drive list as returned by getDVDInfoFromDevTree().
102 */
103typedef struct SOLARISDVD
104{
105 struct SOLARISDVD *pNext;
106 char szDescription[512];
107 char szRawDiskPath[PATH_MAX];
108} SOLARISDVD;
109/** Pointer to a Solaris DVD descriptor. */
110typedef SOLARISDVD *PSOLARISDVD;
111
112
113
114#endif /* RT_OS_SOLARIS */
115
116#ifdef RT_OS_WINDOWS
117# define _WIN32_DCOM
118# include <windows.h>
119# include <shellapi.h>
120# define INITGUID
121# include <guiddef.h>
122# include <devguid.h>
123# include <objbase.h>
124//# include <setupapi.h>
125# include <shlobj.h>
126# include <cfgmgr32.h>
127
128#endif /* RT_OS_WINDOWS */
129
130#ifdef RT_OS_DARWIN
131# include "darwin/iokit.h"
132#endif
133
134#ifdef VBOX_WITH_CROGL
135#include <VBox/VBoxOGL.h>
136#endif /* VBOX_WITH_CROGL */
137
138#include <iprt/asm-amd64-x86.h>
139#include <iprt/string.h>
140#include <iprt/mp.h>
141#include <iprt/time.h>
142#include <iprt/param.h>
143#include <iprt/env.h>
144#include <iprt/mem.h>
145#include <iprt/system.h>
146#ifndef RT_OS_WINDOWS
147# include <iprt/path.h>
148#endif
149#ifdef RT_OS_SOLARIS
150# include <iprt/ctype.h>
151#endif
152#ifdef VBOX_WITH_HOSTNETIF_API
153# include "netif.h"
154#endif
155
156/* XXX Solaris: definitions in /usr/include/sys/regset.h clash with hm_svm.h */
157#undef DS
158#undef ES
159#undef CS
160#undef SS
161#undef FS
162#undef GS
163
164#include <VBox/usb.h>
165#include <VBox/vmm/hm_svm.h>
166#include <VBox/err.h>
167#include <VBox/settings.h>
168#include <VBox/sup.h>
169#include <iprt/x86.h>
170
171#include "VBox/com/MultiResult.h"
172#include "VBox/com/array.h"
173
174#include <stdio.h>
175
176#include <algorithm>
177#include <string>
178#include <vector>
179
180#include "HostDnsService.h"
181
182////////////////////////////////////////////////////////////////////////////////
183//
184// Host private data definition
185//
186////////////////////////////////////////////////////////////////////////////////
187
188struct Host::Data
189{
190 Data()
191 :
192 fDVDDrivesListBuilt(false),
193 fFloppyDrivesListBuilt(false)
194 {};
195
196 VirtualBox *pParent;
197
198 HostNetworkInterfaceList llNetIfs; // list of network interfaces
199
200#ifdef VBOX_WITH_USB
201 USBDeviceFilterList llChildren; // all USB device filters
202 USBDeviceFilterList llUSBDeviceFilters; // USB device filters in use by the USB proxy service
203
204 /** Pointer to the USBProxyService object. */
205 USBProxyService *pUSBProxyService;
206#endif /* VBOX_WITH_USB */
207
208 // list of host drives; lazily created by getDVDDrives() and getFloppyDrives(),
209 // and protected by the medium tree lock handle (including the bools).
210 MediaList llDVDDrives,
211 llFloppyDrives;
212 bool fDVDDrivesListBuilt,
213 fFloppyDrivesListBuilt;
214
215#if defined(RT_OS_LINUX) || defined(RT_OS_FREEBSD)
216 /** Object with information about host drives */
217 VBoxMainDriveInfo hostDrives;
218#endif
219
220 /** @name Features that can be queried with GetProcessorFeature.
221 * @{ */
222 bool fVTSupported,
223 fLongModeSupported,
224 fPAESupported,
225 fNestedPagingSupported,
226 fRecheckVTSupported;
227
228 /** @} */
229
230 /** 3D hardware acceleration supported? Tristate, -1 meaning not probed. */
231 int f3DAccelerationSupported;
232
233 HostPowerService *pHostPowerService;
234 /** Host's DNS informaton fetching */
235 HostDnsMonitorProxy hostDnsMonitorProxy;
236};
237
238
239////////////////////////////////////////////////////////////////////////////////
240//
241// Constructor / destructor
242//
243////////////////////////////////////////////////////////////////////////////////
244DEFINE_EMPTY_CTOR_DTOR(Host)
245
246HRESULT Host::FinalConstruct()
247{
248 return BaseFinalConstruct();
249}
250
251void Host::FinalRelease()
252{
253 uninit();
254 BaseFinalRelease();
255}
256
257/**
258 * Initializes the host object.
259 *
260 * @param aParent VirtualBox parent object.
261 */
262HRESULT Host::init(VirtualBox *aParent)
263{
264 HRESULT hrc;
265 LogFlowThisFunc(("aParent=%p\n", aParent));
266
267 /* Enclose the state transition NotReady->InInit->Ready */
268 AutoInitSpan autoInitSpan(this);
269 AssertReturn(autoInitSpan.isOk(), E_FAIL);
270
271 m = new Data();
272
273 m->pParent = aParent;
274
275#ifdef VBOX_WITH_USB
276 /*
277 * Create and initialize the USB Proxy Service.
278 */
279 m->pUSBProxyService = new USBProxyService(this);
280 hrc = m->pUSBProxyService->init();
281 AssertComRCReturn(hrc, hrc);
282#endif /* VBOX_WITH_USB */
283
284#ifdef VBOX_WITH_RESOURCE_USAGE_API
285 i_registerMetrics(aParent->i_performanceCollector());
286#endif /* VBOX_WITH_RESOURCE_USAGE_API */
287 /* Create the list of network interfaces so their metrics get registered. */
288 i_updateNetIfList();
289
290 m->hostDnsMonitorProxy.init(HostDnsMonitor::getHostDnsMonitor(m->pParent), m->pParent);
291
292#if defined(RT_OS_WINDOWS)
293 m->pHostPowerService = new HostPowerServiceWin(m->pParent);
294#elif defined(RT_OS_LINUX) && defined(VBOX_WITH_DBUS)
295 m->pHostPowerService = new HostPowerServiceLinux(m->pParent);
296#elif defined(RT_OS_DARWIN)
297 m->pHostPowerService = new HostPowerServiceDarwin(m->pParent);
298#else
299 m->pHostPowerService = new HostPowerService(m->pParent);
300#endif
301
302 /* Cache the features reported by GetProcessorFeature. */
303 m->fVTSupported = false;
304 m->fLongModeSupported = false;
305 m->fPAESupported = false;
306 m->fNestedPagingSupported = false;
307 m->fRecheckVTSupported = false;
308
309 if (ASMHasCpuId())
310 {
311 /* Note! This code is duplicated in SUPDrv.c and other places! */
312 uint32_t uMaxId, uVendorEBX, uVendorECX, uVendorEDX;
313 ASMCpuId(0, &uMaxId, &uVendorEBX, &uVendorECX, &uVendorEDX);
314 if (ASMIsValidStdRange(uMaxId))
315 {
316 /* PAE? */
317 uint32_t uDummy, fFeaturesEcx, fFeaturesEdx;
318 ASMCpuId(1, &uDummy, &uDummy, &fFeaturesEcx, &fFeaturesEdx);
319 m->fPAESupported = RT_BOOL(fFeaturesEdx & X86_CPUID_FEATURE_EDX_PAE);
320
321 /* Long Mode? */
322 uint32_t uExtMaxId, fExtFeaturesEcx, fExtFeaturesEdx;
323 ASMCpuId(0x80000000, &uExtMaxId, &uDummy, &uDummy, &uDummy);
324 ASMCpuId(0x80000001, &uDummy, &uDummy, &fExtFeaturesEcx, &fExtFeaturesEdx);
325 m->fLongModeSupported = ASMIsValidExtRange(uExtMaxId)
326 && (fExtFeaturesEdx & X86_CPUID_EXT_FEATURE_EDX_LONG_MODE);
327
328#if defined(RT_OS_DARWIN) && ARCH_BITS == 32 /* darwin.x86 has some optimizations of 64-bit on 32-bit. */
329 int f64bitCapable = 0;
330 size_t cbParameter = sizeof(f64bitCapable);
331 if (sysctlbyname("hw.cpu64bit_capable", &f64bitCapable, &cbParameter, NULL, NULL) != -1)
332 m->fLongModeSupported = f64bitCapable != 0;
333#endif
334
335 /* VT-x? */
336 if ( ASMIsIntelCpuEx(uVendorEBX, uVendorECX, uVendorEDX)
337 || ASMIsViaCentaurCpuEx(uVendorEBX, uVendorECX, uVendorEDX))
338 {
339 if ( (fFeaturesEcx & X86_CPUID_FEATURE_ECX_VMX)
340 && (fFeaturesEdx & X86_CPUID_FEATURE_EDX_MSR)
341 && (fFeaturesEdx & X86_CPUID_FEATURE_EDX_FXSR)
342 )
343 {
344 int rc = SUPR3QueryVTxSupported();
345 if (RT_SUCCESS(rc))
346 m->fVTSupported = true;
347 }
348 }
349 /* AMD-V */
350 else if (ASMIsAmdCpuEx(uVendorEBX, uVendorECX, uVendorEDX))
351 {
352 if ( (fExtFeaturesEcx & X86_CPUID_AMD_FEATURE_ECX_SVM)
353 && (fFeaturesEdx & X86_CPUID_FEATURE_EDX_MSR)
354 && (fFeaturesEdx & X86_CPUID_FEATURE_EDX_FXSR)
355 && ASMIsValidExtRange(uExtMaxId)
356 )
357 {
358 m->fVTSupported = true;
359
360 /* Query AMD features. */
361 if (uExtMaxId >= 0x8000000a)
362 {
363 uint32_t fSVMFeaturesEdx;
364 ASMCpuId(0x8000000a, &uDummy, &uDummy, &uDummy, &fSVMFeaturesEdx);
365 if (fSVMFeaturesEdx & AMD_CPUID_SVM_FEATURE_EDX_NESTED_PAGING)
366 m->fNestedPagingSupported = true;
367 }
368 }
369 }
370 }
371 }
372
373 /* Check with SUPDrv if VT-x and AMD-V are really supported (may fail). */
374 if (m->fVTSupported)
375 {
376 int rc = SUPR3InitEx(false /*fUnrestricted*/, NULL);
377 if (RT_SUCCESS(rc))
378 {
379 uint32_t fVTCaps;
380 rc = SUPR3QueryVTCaps(&fVTCaps);
381 if (RT_SUCCESS(rc))
382 {
383 Assert(fVTCaps & (SUPVTCAPS_AMD_V | SUPVTCAPS_VT_X));
384 if (fVTCaps & SUPVTCAPS_NESTED_PAGING)
385 m->fNestedPagingSupported = true;
386 else
387 Assert(m->fNestedPagingSupported == false);
388 }
389 else
390 {
391 LogRel(("SUPR0QueryVTCaps -> %Rrc\n", rc));
392 m->fVTSupported = m->fNestedPagingSupported = false;
393 }
394 rc = SUPR3Term(false);
395 AssertRC(rc);
396 }
397 else
398 m->fRecheckVTSupported = true; /* Try again later when the driver is loaded. */
399 }
400
401#ifdef VBOX_WITH_CROGL
402 /* Test for 3D hardware acceleration support later when (if ever) need. */
403 m->f3DAccelerationSupported = -1;
404#else
405 m->f3DAccelerationSupported = false;
406#endif
407
408#if defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD)
409 /* Extract the list of configured host-only interfaces */
410 std::set<Utf8Str> aConfiguredNames;
411 SafeArray<BSTR> aGlobalExtraDataKeys;
412 hrc = aParent->GetExtraDataKeys(ComSafeArrayAsOutParam(aGlobalExtraDataKeys));
413 AssertMsg(SUCCEEDED(hrc), ("VirtualBox::GetExtraDataKeys failed with %Rhrc\n", hrc));
414 for (size_t i = 0; i < aGlobalExtraDataKeys.size(); ++i)
415 {
416 Utf8Str strKey = aGlobalExtraDataKeys[i];
417
418 if (!strKey.startsWith("HostOnly/vboxnet"))
419 continue;
420
421 size_t pos = strKey.find("/", sizeof("HostOnly/vboxnet"));
422 if (pos != Utf8Str::npos)
423 aConfiguredNames.insert(strKey.substr(sizeof("HostOnly"),
424 pos - sizeof("HostOnly")));
425 }
426
427 for (std::set<Utf8Str>::const_iterator it = aConfiguredNames.begin();
428 it != aConfiguredNames.end();
429 ++it)
430 {
431 ComPtr<IHostNetworkInterface> hif;
432 ComPtr<IProgress> progress;
433
434 int r = NetIfCreateHostOnlyNetworkInterface(m->pParent,
435 hif.asOutParam(),
436 progress.asOutParam(),
437 it->c_str());
438 if (RT_FAILURE(r))
439 LogRel(("failed to create %s, error (0x%x)\n", it->c_str(), r));
440 }
441
442#endif /* defined(RT_OS_LINUX) || defined(RT_OS_DARWIN) || defined(RT_OS_FREEBSD) */
443
444 /* Confirm a successful initialization */
445 autoInitSpan.setSucceeded();
446
447 return S_OK;
448}
449
450/**
451 * Uninitializes the host object and sets the ready flag to FALSE.
452 * Called either from FinalRelease() or by the parent when it gets destroyed.
453 */
454void Host::uninit()
455{
456 LogFlowThisFunc(("\n"));
457
458 /* Enclose the state transition Ready->InUninit->NotReady */
459 AutoUninitSpan autoUninitSpan(this);
460 if (autoUninitSpan.uninitDone())
461 return;
462
463#ifdef VBOX_WITH_RESOURCE_USAGE_API
464 PerformanceCollector *aCollector = m->pParent->i_performanceCollector();
465 i_unregisterMetrics(aCollector);
466#endif /* VBOX_WITH_RESOURCE_USAGE_API */
467 /*
468 * Note that unregisterMetrics() has unregistered all metrics associated
469 * with Host including network interface ones. We can destroy network
470 * interface objects now. Don't forget the uninit call, otherwise this
471 * causes a race with crashing API clients getting their stale references
472 * cleaned up and VirtualBox shutting down.
473 */
474 while (!m->llNetIfs.empty())
475 {
476 ComObjPtr<HostNetworkInterface> &pNet = m->llNetIfs.front();
477 pNet->uninit();
478 m->llNetIfs.pop_front();
479 }
480
481#ifdef VBOX_WITH_USB
482 /* wait for USB proxy service to terminate before we uninit all USB
483 * devices */
484 LogFlowThisFunc(("Stopping USB proxy service...\n"));
485 delete m->pUSBProxyService;
486 m->pUSBProxyService = NULL;
487 LogFlowThisFunc(("Done stopping USB proxy service.\n"));
488#endif
489
490 delete m->pHostPowerService;
491
492#ifdef VBOX_WITH_USB
493 /* uninit all USB device filters still referenced by clients
494 * Note! HostUSBDeviceFilter::uninit() will modify llChildren.
495 * This list should be already empty, but better be safe than sorry. */
496 while (!m->llChildren.empty())
497 {
498 ComObjPtr<HostUSBDeviceFilter> &pChild = m->llChildren.front();
499 pChild->uninit();
500 m->llChildren.pop_front();
501 }
502
503 /* No need to uninit these, as either Machine::uninit() or the above loop
504 * already covered them all. Subset of llChildren. */
505 m->llUSBDeviceFilters.clear();
506#endif
507
508 /* uninit all host DVD medium objects */
509 while (!m->llDVDDrives.empty())
510 {
511 ComObjPtr<Medium> &pMedium = m->llDVDDrives.front();
512 pMedium->uninit();
513 m->llDVDDrives.pop_front();
514 }
515 /* uninit all host floppy medium objects */
516 while (!m->llFloppyDrives.empty())
517 {
518 ComObjPtr<Medium> &pMedium = m->llFloppyDrives.front();
519 pMedium->uninit();
520 m->llFloppyDrives.pop_front();
521 }
522
523 delete m;
524 m = NULL;
525}
526
527////////////////////////////////////////////////////////////////////////////////
528//
529// IHost public methods
530//
531////////////////////////////////////////////////////////////////////////////////
532
533/**
534 * Returns a list of host DVD drives.
535 *
536 * @returns COM status code
537 * @param drives address of result pointer
538 */
539
540HRESULT Host::getDVDDrives(std::vector<ComPtr<IMedium> > &aDVDDrives)
541{
542 AutoWriteLock treeLock(m->pParent->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
543
544 MediaList *pList;
545 HRESULT rc = i_getDrives(DeviceType_DVD, true /* fRefresh */, pList, treeLock);
546 if (FAILED(rc))
547 return rc;
548
549 aDVDDrives.resize(pList->size());
550 size_t i = 0;
551 for (MediaList::const_iterator it = pList->begin(); it != pList->end(); ++it, ++i)
552 (*it).queryInterfaceTo(aDVDDrives[i].asOutParam());
553
554 return S_OK;
555}
556
557/**
558 * Returns a list of host floppy drives.
559 *
560 * @returns COM status code
561 * @param drives address of result pointer
562 */
563HRESULT Host::getFloppyDrives(std::vector<ComPtr<IMedium> > &aFloppyDrives)
564{
565 AutoWriteLock treeLock(m->pParent->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
566
567 MediaList *pList;
568 HRESULT rc = i_getDrives(DeviceType_Floppy, true /* fRefresh */, pList, treeLock);
569 if (FAILED(rc))
570 return rc;
571
572 aFloppyDrives.resize(pList->size());
573 size_t i = 0;
574 for (MediaList::const_iterator it = pList->begin(); it != pList->end(); ++it, ++i)
575 (*it).queryInterfaceTo(aFloppyDrives[i].asOutParam());
576
577 return S_OK;
578}
579
580
581#if defined(RT_OS_WINDOWS) && defined(VBOX_WITH_NETFLT)
582# define VBOX_APP_NAME L"VirtualBox"
583
584static int vboxNetWinAddComponent(std::list< ComObjPtr<HostNetworkInterface> > *pPist,
585 INetCfgComponent *pncc)
586{
587 LPWSTR lpszName;
588 GUID IfGuid;
589 HRESULT hr;
590 int rc = VERR_GENERAL_FAILURE;
591
592 hr = pncc->GetDisplayName(&lpszName);
593 Assert(hr == S_OK);
594 if (hr == S_OK)
595 {
596 Bstr name((CBSTR)lpszName);
597
598 hr = pncc->GetInstanceGuid(&IfGuid);
599 Assert(hr == S_OK);
600 if (hr == S_OK)
601 {
602 /* create a new object and add it to the list */
603 ComObjPtr<HostNetworkInterface> iface;
604 iface.createObject();
605 /* remove the curly bracket at the end */
606 if (SUCCEEDED(iface->init(name, name, Guid(IfGuid), HostNetworkInterfaceType_Bridged)))
607 {
608// iface->setVirtualBox(m->pParent);
609 pPist->push_back(iface);
610 rc = VINF_SUCCESS;
611 }
612 else
613 {
614 Assert(0);
615 }
616 }
617 CoTaskMemFree(lpszName);
618 }
619
620 return rc;
621}
622#endif /* defined(RT_OS_WINDOWS) && defined(VBOX_WITH_NETFLT) */
623
624/**
625 * Returns a list of host network interfaces.
626 *
627 * @returns COM status code
628 * @param drives address of result pointer
629 */
630HRESULT Host::getNetworkInterfaces(std::vector<ComPtr<IHostNetworkInterface> > &aNetworkInterfaces)
631{
632#if defined(RT_OS_WINDOWS) || defined(VBOX_WITH_NETFLT) /*|| defined(RT_OS_OS2)*/
633# ifdef VBOX_WITH_HOSTNETIF_API
634 HRESULT rc = i_updateNetIfList();
635 if (FAILED(rc))
636 {
637 Log(("Failed to update host network interface list with rc=%Rhrc\n", rc));
638 return rc;
639 }
640
641 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
642
643 aNetworkInterfaces.resize(m->llNetIfs.size());
644 size_t i = 0;
645 for (HostNetworkInterfaceList::iterator it = m->llNetIfs.begin(); it != m->llNetIfs.end(); ++it, ++i)
646 (*it).queryInterfaceTo(aNetworkInterfaces[i].asOutParam());
647
648 return S_OK;
649# else
650 std::list<ComObjPtr<HostNetworkInterface> > list;
651
652# if defined(RT_OS_DARWIN)
653 PDARWINETHERNIC pEtherNICs = DarwinGetEthernetControllers();
654 while (pEtherNICs)
655 {
656 ComObjPtr<HostNetworkInterface> IfObj;
657 IfObj.createObject();
658 if (SUCCEEDED(IfObj->init(Bstr(pEtherNICs->szName), Guid(pEtherNICs->Uuid), HostNetworkInterfaceType_Bridged)))
659 list.push_back(IfObj);
660
661 /* next, free current */
662 void *pvFree = pEtherNICs;
663 pEtherNICs = pEtherNICs->pNext;
664 RTMemFree(pvFree);
665 }
666
667# elif defined RT_OS_WINDOWS
668# ifndef VBOX_WITH_NETFLT
669 hr = E_NOTIMPL;
670# else /* # if defined VBOX_WITH_NETFLT */
671 INetCfg *pNc;
672 INetCfgComponent *pMpNcc;
673 INetCfgComponent *pTcpIpNcc;
674 LPWSTR lpszApp;
675 HRESULT hr;
676 IEnumNetCfgBindingPath *pEnumBp;
677 INetCfgBindingPath *pBp;
678 IEnumNetCfgBindingInterface *pEnumBi;
679 INetCfgBindingInterface *pBi;
680
681 /* we are using the INetCfg API for getting the list of miniports */
682 hr = VBoxNetCfgWinQueryINetCfg(FALSE,
683 VBOX_APP_NAME,
684 &pNc,
685 &lpszApp);
686 Assert(hr == S_OK);
687 if (hr == S_OK)
688 {
689# ifdef VBOX_NETFLT_ONDEMAND_BIND
690 /* for the protocol-based approach for now we just get all miniports the MS_TCPIP protocol binds to */
691 hr = pNc->FindComponent(L"MS_TCPIP", &pTcpIpNcc);
692# else
693 /* for the filter-based approach we get all miniports our filter (oracle_VBoxNetLwf)is bound to */
694 hr = pNc->FindComponent(L"oracle_VBoxNetLwf", &pTcpIpNcc);
695 if (hr != S_OK)
696 {
697 /* fall back to NDIS5 miniport lookup (sun_VBoxNetFlt) */
698 hr = pNc->FindComponent(L"sun_VBoxNetFlt", &pTcpIpNcc);
699 }
700# ifndef VBOX_WITH_HARDENING
701 if (hr != S_OK)
702 {
703 /* TODO: try to install the netflt from here */
704 }
705# endif
706
707# endif
708
709 if (hr == S_OK)
710 {
711 hr = VBoxNetCfgWinGetBindingPathEnum(pTcpIpNcc, EBP_BELOW, &pEnumBp);
712 Assert(hr == S_OK);
713 if (hr == S_OK)
714 {
715 hr = VBoxNetCfgWinGetFirstBindingPath(pEnumBp, &pBp);
716 Assert(hr == S_OK || hr == S_FALSE);
717 while (hr == S_OK)
718 {
719 /* S_OK == enabled, S_FALSE == disabled */
720 if (pBp->IsEnabled() == S_OK)
721 {
722 hr = VBoxNetCfgWinGetBindingInterfaceEnum(pBp, &pEnumBi);
723 Assert(hr == S_OK);
724 if (hr == S_OK)
725 {
726 hr = VBoxNetCfgWinGetFirstBindingInterface(pEnumBi, &pBi);
727 Assert(hr == S_OK);
728 while (hr == S_OK)
729 {
730 hr = pBi->GetLowerComponent(&pMpNcc);
731 Assert(hr == S_OK);
732 if (hr == S_OK)
733 {
734 ULONG uComponentStatus;
735 hr = pMpNcc->GetDeviceStatus(&uComponentStatus);
736 Assert(hr == S_OK);
737 if (hr == S_OK)
738 {
739 if (uComponentStatus == 0)
740 {
741 vboxNetWinAddComponent(&list, pMpNcc);
742 }
743 }
744 VBoxNetCfgWinReleaseRef(pMpNcc);
745 }
746 VBoxNetCfgWinReleaseRef(pBi);
747
748 hr = VBoxNetCfgWinGetNextBindingInterface(pEnumBi, &pBi);
749 }
750 VBoxNetCfgWinReleaseRef(pEnumBi);
751 }
752 }
753 VBoxNetCfgWinReleaseRef(pBp);
754
755 hr = VBoxNetCfgWinGetNextBindingPath(pEnumBp, &pBp);
756 }
757 VBoxNetCfgWinReleaseRef(pEnumBp);
758 }
759 VBoxNetCfgWinReleaseRef(pTcpIpNcc);
760 }
761 else
762 {
763 LogRel(("failed to get the oracle_VBoxNetLwf(sun_VBoxNetFlt) component, error (0x%x)\n", hr));
764 }
765
766 VBoxNetCfgWinReleaseINetCfg(pNc, FALSE);
767 }
768# endif /* # if defined VBOX_WITH_NETFLT */
769
770
771# elif defined RT_OS_LINUX
772 int sock = socket(AF_INET, SOCK_DGRAM, 0);
773 if (sock >= 0)
774 {
775 char pBuffer[2048];
776 struct ifconf ifConf;
777 ifConf.ifc_len = sizeof(pBuffer);
778 ifConf.ifc_buf = pBuffer;
779 if (ioctl(sock, SIOCGIFCONF, &ifConf) >= 0)
780 {
781 for (struct ifreq *pReq = ifConf.ifc_req; (char*)pReq < pBuffer + ifConf.ifc_len; pReq++)
782 {
783 if (ioctl(sock, SIOCGIFHWADDR, pReq) >= 0)
784 {
785 if (pReq->ifr_hwaddr.sa_family == ARPHRD_ETHER)
786 {
787 RTUUID uuid;
788 Assert(sizeof(uuid) <= sizeof(*pReq));
789 memcpy(&uuid, pReq, sizeof(uuid));
790
791 ComObjPtr<HostNetworkInterface> IfObj;
792 IfObj.createObject();
793 if (SUCCEEDED(IfObj->init(Bstr(pReq->ifr_name), Guid(uuid), HostNetworkInterfaceType_Bridged)))
794 list.push_back(IfObj);
795 }
796 }
797 }
798 }
799 close(sock);
800 }
801# endif /* RT_OS_LINUX */
802
803 aNetworkInterfaces.resize(list.size());
804 size_t i = 0;
805 for (std::list<ComObjPtr<HostNetworkInterface> >::const_iterator it = list.begin(); it != list.end(); ++it, ++i)
806 aNetworkInterfaces[i] = *it;
807
808 return S_OK;
809# endif
810#else
811 /* Not implemented / supported on this platform. */
812 ReturnComNotImplemented();
813#endif
814}
815
816HRESULT Host::getUSBDevices(std::vector<ComPtr<IHostUSBDevice> > &aUSBDevices)
817{
818#ifdef VBOX_WITH_USB
819 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
820
821 MultiResult rc = i_checkUSBProxyService();
822 if (FAILED(rc))
823 return rc;
824
825 return m->pUSBProxyService->getDeviceCollection(aUSBDevices);
826#else
827 /* Note: The GUI depends on this method returning E_NOTIMPL with no
828 * extended error info to indicate that USB is simply not available
829 * (w/o treating it as a failure), for example, as in OSE. */
830 NOREF(aUSBDevices);
831# ifndef RT_OS_WINDOWS
832 NOREF(aUSBDevices);
833# endif
834 ReturnComNotImplemented();
835#endif
836}
837
838/**
839 * This method return the list of registered name servers
840 */
841HRESULT Host::getNameServers(std::vector<com::Utf8Str> &aNameServers)
842{
843 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
844 return m->hostDnsMonitorProxy.GetNameServers(aNameServers);
845}
846
847
848/**
849 * This method returns the domain name of the host
850 */
851HRESULT Host::getDomainName(com::Utf8Str &aDomainName)
852{
853 /* XXX: note here should be synchronization with thread polling state
854 * changes in name resoving system on host */
855 return m->hostDnsMonitorProxy.GetDomainName(&aDomainName);
856}
857
858
859/**
860 * This method returns the search string.
861 */
862HRESULT Host::getSearchStrings(std::vector<com::Utf8Str> &aSearchStrings)
863{
864 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
865 return m->hostDnsMonitorProxy.GetSearchStrings(aSearchStrings);
866}
867
868HRESULT Host::getUSBDeviceFilters(std::vector<ComPtr<IHostUSBDeviceFilter> > &aUSBDeviceFilters)
869{
870#ifdef VBOX_WITH_USB
871 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
872
873 MultiResult rc = i_checkUSBProxyService();
874 if (FAILED(rc))
875 return rc;
876
877 aUSBDeviceFilters.resize(m->llUSBDeviceFilters.size());
878 size_t i = 0;
879 for (USBDeviceFilterList::iterator it = m->llUSBDeviceFilters.begin(); it != m->llUSBDeviceFilters.end(); ++it, ++i)
880 (*it).queryInterfaceTo(aUSBDeviceFilters[i].asOutParam());
881
882 return rc;
883#else
884 /* Note: The GUI depends on this method returning E_NOTIMPL with no
885 * extended error info to indicate that USB is simply not available
886 * (w/o treating it as a failure), for example, as in OSE. */
887 NOREF(aUSBDeviceFilters);
888# ifndef RT_OS_WINDOWS
889 NOREF(aUSBDeviceFilters);
890# endif
891 ReturnComNotImplemented();
892#endif
893}
894
895/**
896 * Returns the number of installed logical processors
897 *
898 * @returns COM status code
899 * @param count address of result variable
900 */
901
902HRESULT Host::getProcessorCount(ULONG *aCount)
903{
904 // no locking required
905
906 *aCount = RTMpGetPresentCount();
907 return S_OK;
908}
909
910/**
911 * Returns the number of online logical processors
912 *
913 * @returns COM status code
914 * @param count address of result variable
915 */
916HRESULT Host::getProcessorOnlineCount(ULONG *aCount)
917{
918 // no locking required
919
920 *aCount = RTMpGetOnlineCount();
921 return S_OK;
922}
923
924/**
925 * Returns the number of installed physical processor cores.
926 *
927 * @returns COM status code
928 * @param count address of result variable
929 */
930HRESULT Host::getProcessorCoreCount(ULONG *aCount)
931{
932 // no locking required
933
934 *aCount = RTMpGetPresentCoreCount();
935 return S_OK;
936}
937
938/**
939 * Returns the number of installed physical processor cores.
940 *
941 * @returns COM status code
942 * @param count address of result variable
943 */
944HRESULT Host::getProcessorOnlineCoreCount(ULONG *aCount)
945{
946 // no locking required
947
948 *aCount = RTMpGetOnlineCoreCount();
949 return S_OK;
950}
951
952/**
953 * Returns the (approximate) maximum speed of the given host CPU in MHz
954 *
955 * @returns COM status code
956 * @param cpu id to get info for.
957 * @param speed address of result variable, speed is 0 if unknown or aCpuId is invalid.
958 */
959HRESULT Host::getProcessorSpeed(ULONG aCpuId,
960 ULONG *aSpeed)
961{
962 // no locking required
963
964 *aSpeed = RTMpGetMaxFrequency(aCpuId);
965 return S_OK;
966}
967
968/**
969 * Returns a description string for the host CPU
970 *
971 * @returns COM status code
972 * @param cpu id to get info for.
973 * @param description address of result variable, empty string if not known or aCpuId is invalid.
974 */
975HRESULT Host::getProcessorDescription(ULONG aCpuId, com::Utf8Str &aDescription)
976{
977 // no locking required
978
979 char szCPUModel[80];
980 szCPUModel[0] = 0;
981 int vrc = RTMpGetDescription(aCpuId, szCPUModel, sizeof(szCPUModel));
982 if (RT_FAILURE(vrc))
983 return E_FAIL; /** @todo error reporting? */
984
985 aDescription = Utf8Str(szCPUModel);
986
987 return S_OK;
988}
989
990/**
991 * Returns whether a host processor feature is supported or not
992 *
993 * @returns COM status code
994 * @param Feature to query.
995 * @param address of supported bool result variable
996 */
997HRESULT Host::getProcessorFeature(ProcessorFeature_T aFeature, BOOL *aSupported)
998{
999 /* Validate input. */
1000 switch (aFeature)
1001 {
1002 case ProcessorFeature_HWVirtEx:
1003 case ProcessorFeature_PAE:
1004 case ProcessorFeature_LongMode:
1005 case ProcessorFeature_NestedPaging:
1006 break;
1007 default:
1008 return setError(E_INVALIDARG, tr("The aFeature value %d (%#x) is out of range."), (int)aFeature, (int)aFeature);
1009 }
1010
1011 /* Do the job. */
1012 AutoCaller autoCaller(this);
1013 HRESULT hrc = autoCaller.rc();
1014 if (SUCCEEDED(hrc))
1015 {
1016 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1017
1018 if ( m->fRecheckVTSupported
1019 && ( aFeature == ProcessorFeature_HWVirtEx
1020 || aFeature == ProcessorFeature_NestedPaging)
1021 )
1022 {
1023 alock.release();
1024
1025 /* Perhaps the driver is available now... */
1026 int rc = SUPR3InitEx(false /*fUnrestricted*/, NULL);
1027 if (RT_SUCCESS(rc))
1028 {
1029 uint32_t fVTCaps;
1030 rc = SUPR3QueryVTCaps(&fVTCaps);
1031
1032 AutoWriteLock wlock(this COMMA_LOCKVAL_SRC_POS);
1033 if (RT_SUCCESS(rc))
1034 {
1035 Assert(fVTCaps & (SUPVTCAPS_AMD_V | SUPVTCAPS_VT_X));
1036 if (fVTCaps & SUPVTCAPS_NESTED_PAGING)
1037 m->fNestedPagingSupported = true;
1038 else
1039 Assert(m->fNestedPagingSupported == false);
1040 }
1041 else
1042 {
1043 LogRel(("SUPR0QueryVTCaps -> %Rrc\n", rc));
1044 m->fVTSupported = m->fNestedPagingSupported = true;
1045 }
1046 rc = SUPR3Term(false);
1047 AssertRC(rc);
1048 m->fRecheckVTSupported = false; /* No need to try again, we cached everything. */
1049 }
1050
1051 alock.acquire();
1052 }
1053
1054 switch (aFeature)
1055 {
1056 case ProcessorFeature_HWVirtEx:
1057 *aSupported = m->fVTSupported;
1058 break;
1059
1060 case ProcessorFeature_PAE:
1061 *aSupported = m->fPAESupported;
1062 break;
1063
1064 case ProcessorFeature_LongMode:
1065 *aSupported = m->fLongModeSupported;
1066 break;
1067
1068 case ProcessorFeature_NestedPaging:
1069 *aSupported = m->fNestedPagingSupported;
1070 break;
1071
1072 default:
1073 AssertFailed();
1074 }
1075 }
1076 return hrc;
1077}
1078
1079/**
1080 * Returns the specific CPUID leaf.
1081 *
1082 * @returns COM status code
1083 * @param aCpuId The CPU number. Mostly ignored.
1084 * @param aLeaf The leaf number.
1085 * @param aSubLeaf The sub-leaf number.
1086 * @param aValEAX Where to return EAX.
1087 * @param aValEBX Where to return EBX.
1088 * @param aValECX Where to return ECX.
1089 * @param aValEDX Where to return EDX.
1090 */
1091HRESULT Host::getProcessorCPUIDLeaf(ULONG aCpuId, ULONG aLeaf, ULONG aSubLeaf,
1092 ULONG *aValEAX, ULONG *aValEBX, ULONG *aValECX, ULONG *aValEDX)
1093{
1094 // no locking required
1095
1096 /* Check that the CPU is online. */
1097 /** @todo later use RTMpOnSpecific. */
1098 if (!RTMpIsCpuOnline(aCpuId))
1099 return RTMpIsCpuPresent(aCpuId)
1100 ? setError(E_FAIL, tr("CPU no.%u is not present"), aCpuId)
1101 : setError(E_FAIL, tr("CPU no.%u is not online"), aCpuId);
1102
1103 uint32_t uEAX, uEBX, uECX, uEDX;
1104 ASMCpuId_Idx_ECX(aLeaf, aSubLeaf, &uEAX, &uEBX, &uECX, &uEDX);
1105 *aValEAX = uEAX;
1106 *aValEBX = uEBX;
1107 *aValECX = uECX;
1108 *aValEDX = uEDX;
1109
1110 return S_OK;
1111}
1112
1113/**
1114 * Returns the amount of installed system memory in megabytes
1115 *
1116 * @returns COM status code
1117 * @param size address of result variable
1118 */
1119HRESULT Host::getMemorySize(ULONG *aSize)
1120{
1121 // no locking required
1122
1123 uint64_t cb;
1124 int rc = RTSystemQueryTotalRam(&cb);
1125 if (RT_FAILURE(rc))
1126 return E_FAIL;
1127 *aSize = (ULONG)(cb / _1M);
1128 return S_OK;
1129}
1130
1131/**
1132 * Returns the current system memory free space in megabytes
1133 *
1134 * @returns COM status code
1135 * @param available address of result variable
1136 */
1137HRESULT Host::getMemoryAvailable(ULONG *aAvailable)
1138{
1139 // no locking required
1140
1141 uint64_t cb;
1142 int rc = RTSystemQueryAvailableRam(&cb);
1143 if (RT_FAILURE(rc))
1144 return E_FAIL;
1145 *aAvailable = (ULONG)(cb / _1M);
1146 return S_OK;
1147}
1148
1149/**
1150 * Returns the name string of the host operating system
1151 *
1152 * @returns COM status code
1153 * @param os address of result variable
1154 */
1155HRESULT Host::getOperatingSystem(com::Utf8Str &aOperatingSystem)
1156{
1157 // no locking required
1158
1159 char szOSName[80];
1160 int vrc = RTSystemQueryOSInfo(RTSYSOSINFO_PRODUCT, szOSName, sizeof(szOSName));
1161 if (RT_FAILURE(vrc))
1162 return E_FAIL; /** @todo error reporting? */
1163 aOperatingSystem = Utf8Str(szOSName);
1164 return S_OK;
1165}
1166
1167/**
1168 * Returns the version string of the host operating system
1169 *
1170 * @returns COM status code
1171 * @param os address of result variable
1172 */
1173HRESULT Host::getOSVersion(com::Utf8Str &aVersion)
1174{
1175 // no locking required
1176
1177 /* Get the OS release. Reserve some buffer space for the service pack. */
1178 char szOSRelease[128];
1179 int vrc = RTSystemQueryOSInfo(RTSYSOSINFO_RELEASE, szOSRelease, sizeof(szOSRelease) - 32);
1180 if (RT_FAILURE(vrc))
1181 return E_FAIL; /** @todo error reporting? */
1182
1183 /* Append the service pack if present. */
1184 char szOSServicePack[80];
1185 vrc = RTSystemQueryOSInfo(RTSYSOSINFO_SERVICE_PACK, szOSServicePack, sizeof(szOSServicePack));
1186 if (RT_FAILURE(vrc))
1187 {
1188 if (vrc != VERR_NOT_SUPPORTED)
1189 return E_FAIL; /** @todo error reporting? */
1190 szOSServicePack[0] = '\0';
1191 }
1192 if (szOSServicePack[0] != '\0')
1193 {
1194 char *psz = strchr(szOSRelease, '\0');
1195 RTStrPrintf(psz, &szOSRelease[sizeof(szOSRelease)] - psz, "sp%s", szOSServicePack);
1196 }
1197
1198 aVersion = szOSRelease;
1199 return S_OK;
1200}
1201
1202/**
1203 * Returns the current host time in milliseconds since 1970-01-01 UTC.
1204 *
1205 * @returns COM status code
1206 * @param time address of result variable
1207 */
1208HRESULT Host::getUTCTime(LONG64 *aUTCTime)
1209{
1210 // no locking required
1211
1212 RTTIMESPEC now;
1213 *aUTCTime = RTTimeSpecGetMilli(RTTimeNow(&now));
1214
1215 return S_OK;
1216}
1217
1218
1219HRESULT Host::getAcceleration3DAvailable(BOOL *aSupported)
1220{
1221 HRESULT hrc = S_OK;
1222 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1223 if (m->f3DAccelerationSupported != -1)
1224 *aSupported = m->f3DAccelerationSupported;
1225 else
1226 {
1227 alock.release();
1228
1229#ifdef VBOX_WITH_CROGL
1230 bool fSupported = VBoxOglIs3DAccelerationSupported();
1231#else
1232 bool fSupported = false; /* shoudn't get here, but just in case. */
1233#endif
1234 AutoWriteLock alock2(this COMMA_LOCKVAL_SRC_POS);
1235
1236 m->f3DAccelerationSupported = fSupported;
1237 alock2.release();
1238 *aSupported = fSupported;
1239 }
1240
1241#ifdef DEBUG_misha
1242 AssertMsgFailed(("should not be here any more!\n"));
1243#endif
1244
1245 return hrc;
1246}
1247
1248HRESULT Host::createHostOnlyNetworkInterface(ComPtr<IHostNetworkInterface> &aHostInterface,
1249 ComPtr<IProgress> &aProgress)
1250
1251{
1252#ifdef VBOX_WITH_HOSTNETIF_API
1253 /* No need to lock anything. If there ever will - watch out, the function
1254 * called below grabs the VirtualBox lock. */
1255
1256 int r = NetIfCreateHostOnlyNetworkInterface(m->pParent, aHostInterface.asOutParam(), aProgress.asOutParam());
1257 if (RT_SUCCESS(r))
1258 {
1259 if (aHostInterface.isNull())
1260 return setError(E_FAIL,
1261 tr("Unable to create a host network interface"));
1262
1263#if !defined(RT_OS_WINDOWS)
1264 Bstr tmpAddr, tmpMask, tmpName;
1265 HRESULT hrc;
1266 hrc = aHostInterface->COMGETTER(Name)(tmpName.asOutParam());
1267 ComAssertComRCRet(hrc, hrc);
1268 hrc = aHostInterface->COMGETTER(IPAddress)(tmpAddr.asOutParam());
1269 ComAssertComRCRet(hrc, hrc);
1270 hrc = aHostInterface->COMGETTER(NetworkMask)(tmpMask.asOutParam());
1271 ComAssertComRCRet(hrc, hrc);
1272 /*
1273 * We need to write the default IP address and mask to extra data now,
1274 * so the interface gets re-created after vboxnetadp.ko reload.
1275 * Note that we avoid calling EnableStaticIpConfig since it would
1276 * change the address on host's interface as well and we want to
1277 * postpone the change until VM actually starts.
1278 */
1279 hrc = m->pParent->SetExtraData(BstrFmt("HostOnly/%ls/IPAddress",
1280 tmpName.raw()).raw(),
1281 tmpAddr.raw());
1282 ComAssertComRCRet(hrc, hrc);
1283
1284 hrc = m->pParent->SetExtraData(BstrFmt("HostOnly/%ls/IPNetMask",
1285 tmpName.raw()).raw(),
1286 tmpMask.raw());
1287 ComAssertComRCRet(hrc, hrc);
1288#endif
1289 }
1290
1291 return S_OK;
1292#else
1293 return E_NOTIMPL;
1294#endif
1295}
1296
1297HRESULT Host::removeHostOnlyNetworkInterface(const com::Guid &aId,
1298 ComPtr<IProgress> &aProgress)
1299
1300{
1301#ifdef VBOX_WITH_HOSTNETIF_API
1302 /* No need to lock anything, the code below does not touch the state
1303 * of the host object. If that ever changes then check for lock order
1304 * violations with the called functions. */
1305
1306 Bstr name;
1307 HRESULT rc;
1308
1309 /* first check whether an interface with the given name already exists */
1310 {
1311 ComPtr<IHostNetworkInterface> iface;
1312 rc = findHostNetworkInterfaceById(aId, iface);
1313 if (FAILED(rc))
1314 return setError(VBOX_E_OBJECT_NOT_FOUND,
1315 tr("Host network interface with UUID {%RTuuid} does not exist"),
1316 Guid(aId).raw());
1317 rc = iface->COMGETTER(Name)(name.asOutParam());
1318 ComAssertComRCRet(rc, rc);
1319 }
1320
1321 int r = NetIfRemoveHostOnlyNetworkInterface(m->pParent, Guid(aId).ref(), aProgress.asOutParam());
1322 if (RT_SUCCESS(r))
1323 {
1324 /* Drop configuration parameters for removed interface */
1325 rc = m->pParent->SetExtraData(BstrFmt("HostOnly/%ls/IPAddress", name.raw()).raw(), NULL);
1326 rc = m->pParent->SetExtraData(BstrFmt("HostOnly/%ls/IPNetMask", name.raw()).raw(), NULL);
1327 rc = m->pParent->SetExtraData(BstrFmt("HostOnly/%ls/IPV6Address", name.raw()).raw(), NULL);
1328 rc = m->pParent->SetExtraData(BstrFmt("HostOnly/%ls/IPV6NetMask", name.raw()).raw(), NULL);
1329
1330 return S_OK;
1331 }
1332
1333 return r == VERR_NOT_IMPLEMENTED ? E_NOTIMPL : E_FAIL;
1334#else
1335 return E_NOTIMPL;
1336#endif
1337}
1338
1339HRESULT Host::createUSBDeviceFilter(const com::Utf8Str &aName,
1340 ComPtr<IHostUSBDeviceFilter> &aFilter)
1341{
1342#ifdef VBOX_WITH_USB
1343
1344 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1345
1346 ComObjPtr<HostUSBDeviceFilter> filter;
1347 filter.createObject();
1348 HRESULT rc = filter->init(this, Bstr(aName).raw());
1349 ComAssertComRCRet(rc, rc);
1350 rc = filter.queryInterfaceTo(aFilter.asOutParam());
1351 AssertComRCReturn(rc, rc);
1352 return S_OK;
1353#else
1354 /* Note: The GUI depends on this method returning E_NOTIMPL with no
1355 * extended error info to indicate that USB is simply not available
1356 * (w/o treating it as a failure), for example, as in OSE. */
1357 NOREF(aName);
1358 NOREF(aFilter);
1359 ReturnComNotImplemented();
1360#endif
1361}
1362
1363HRESULT Host::insertUSBDeviceFilter(ULONG aPosition,
1364 const ComPtr<IHostUSBDeviceFilter> &aFilter)
1365{
1366#ifdef VBOX_WITH_USB
1367 /* Note: HostUSBDeviceFilter and USBProxyService also uses this lock. */
1368
1369 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1370
1371 MultiResult rc = i_checkUSBProxyService();
1372 if (FAILED(rc))
1373 return rc;
1374
1375 ComObjPtr<HostUSBDeviceFilter> pFilter;
1376 for (USBDeviceFilterList::iterator it = m->llChildren.begin();
1377 it != m->llChildren.end();
1378 ++it)
1379 {
1380 if (*it == aFilter)
1381 {
1382 pFilter = *it;
1383 break;
1384 }
1385 }
1386 if (pFilter.isNull())
1387 return setError(VBOX_E_INVALID_OBJECT_STATE,
1388 tr("The given USB device filter is not created within this VirtualBox instance"));
1389
1390 if (pFilter->mInList)
1391 return setError(E_INVALIDARG,
1392 tr("The given USB device filter is already in the list"));
1393
1394 /* iterate to the position... */
1395 USBDeviceFilterList::iterator itPos = m->llUSBDeviceFilters.begin();
1396 std::advance(itPos, aPosition);
1397 /* ...and insert */
1398 m->llUSBDeviceFilters.insert(itPos, pFilter);
1399 pFilter->mInList = true;
1400
1401 /* notify the proxy (only when the filter is active) */
1402 if ( m->pUSBProxyService->isActive()
1403 && pFilter->i_getData().mActive)
1404 {
1405 ComAssertRet(pFilter->i_getId() == NULL, E_FAIL);
1406 pFilter->i_getId() = m->pUSBProxyService->insertFilter(&pFilter->i_getData().mUSBFilter);
1407 }
1408
1409 // save the global settings; for that we should hold only the VirtualBox lock
1410 alock.release();
1411 AutoWriteLock vboxLock(m->pParent COMMA_LOCKVAL_SRC_POS);
1412 return rc = m->pParent->i_saveSettings();
1413#else
1414
1415 /* Note: The GUI depends on this method returning E_NOTIMPL with no
1416 * extended error info to indicate that USB is simply not available
1417 * (w/o treating it as a failure), for example, as in OSE. */
1418 NOREF(aPosition);
1419 NOREF(aFilter);
1420 ReturnComNotImplemented();
1421#endif
1422}
1423
1424HRESULT Host::removeUSBDeviceFilter(ULONG aPosition)
1425{
1426#ifdef VBOX_WITH_USB
1427
1428 /* Note: HostUSBDeviceFilter and USBProxyService also uses this lock. */
1429 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1430
1431 MultiResult rc = i_checkUSBProxyService();
1432 if (FAILED(rc))
1433 return rc;
1434
1435 if (!m->llUSBDeviceFilters.size())
1436 return setError(E_INVALIDARG,
1437 tr("The USB device filter list is empty"));
1438
1439 if (aPosition >= m->llUSBDeviceFilters.size())
1440 return setError(E_INVALIDARG,
1441 tr("Invalid position: %lu (must be in range [0, %lu])"),
1442 aPosition, m->llUSBDeviceFilters.size() - 1);
1443
1444 ComObjPtr<HostUSBDeviceFilter> filter;
1445 {
1446 /* iterate to the position... */
1447 USBDeviceFilterList::iterator it = m->llUSBDeviceFilters.begin();
1448 std::advance(it, aPosition);
1449 /* ...get an element from there... */
1450 filter = *it;
1451 /* ...and remove */
1452 filter->mInList = false;
1453 m->llUSBDeviceFilters.erase(it);
1454 }
1455
1456 /* notify the proxy (only when the filter is active) */
1457 if (m->pUSBProxyService->isActive() && filter->i_getData().mActive)
1458 {
1459 ComAssertRet(filter->i_getId() != NULL, E_FAIL);
1460 m->pUSBProxyService->removeFilter(filter->i_getId());
1461 filter->i_getId() = NULL;
1462 }
1463
1464 // save the global settings; for that we should hold only the VirtualBox lock
1465 alock.release();
1466 AutoWriteLock vboxLock(m->pParent COMMA_LOCKVAL_SRC_POS);
1467 return rc = m->pParent->i_saveSettings();
1468#else
1469 /* Note: The GUI depends on this method returning E_NOTIMPL with no
1470 * extended error info to indicate that USB is simply not available
1471 * (w/o treating it as a failure), for example, as in OSE. */
1472 NOREF(aPosition);
1473 ReturnComNotImplemented();
1474#endif
1475}
1476
1477HRESULT Host::findHostDVDDrive(const com::Utf8Str &aName,
1478 ComPtr<IMedium> &aDrive)
1479{
1480 ComObjPtr<Medium> medium;
1481 HRESULT rc = i_findHostDriveByNameOrId(DeviceType_DVD, aName, medium);
1482 if (SUCCEEDED(rc))
1483 rc = medium.queryInterfaceTo(aDrive.asOutParam());
1484 else
1485 rc = setError(rc, Medium::tr("The host DVD drive named '%s' could not be found"), aName.c_str());
1486 return rc;
1487}
1488
1489HRESULT Host::findHostFloppyDrive(const com::Utf8Str &aName, ComPtr<IMedium> &aDrive)
1490{
1491 aDrive = NULL;
1492
1493 ComObjPtr<Medium>medium;
1494
1495 HRESULT rc = i_findHostDriveByNameOrId(DeviceType_Floppy, aName, medium);
1496 if (SUCCEEDED(rc))
1497 return medium.queryInterfaceTo(aDrive.asOutParam());
1498 else
1499 return setError(rc, Medium::tr("The host floppy drive named '%s' could not be found"), aName.c_str());
1500}
1501
1502HRESULT Host::findHostNetworkInterfaceByName(const com::Utf8Str &aName,
1503 ComPtr<IHostNetworkInterface> &aNetworkInterface)
1504{
1505#ifndef VBOX_WITH_HOSTNETIF_API
1506 return E_NOTIMPL;
1507#else
1508 if (!aName.length())
1509 return E_INVALIDARG;
1510
1511 HRESULT rc = i_updateNetIfList();
1512 if (FAILED(rc))
1513 {
1514 Log(("Failed to update host network interface list with rc=%Rhrc\n", rc));
1515 return rc;
1516 }
1517
1518 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1519
1520 ComObjPtr<HostNetworkInterface> found;
1521 for (HostNetworkInterfaceList::iterator it = m->llNetIfs.begin(); it != m->llNetIfs.end(); ++it)
1522 {
1523 Bstr n;
1524 (*it)->COMGETTER(Name)(n.asOutParam());
1525 if (n == aName)
1526 found = *it;
1527 }
1528
1529 if (!found)
1530 return setError(E_INVALIDARG,
1531 HostNetworkInterface::tr("The host network interface named '%s' could not be found"), aName.c_str());
1532
1533 return found.queryInterfaceTo(aNetworkInterface.asOutParam());
1534#endif
1535}
1536
1537HRESULT Host::findHostNetworkInterfaceById(const com::Guid &aId,
1538 ComPtr<IHostNetworkInterface> &aNetworkInterface)
1539{
1540#ifndef VBOX_WITH_HOSTNETIF_API
1541 return E_NOTIMPL;
1542#else
1543 if (!aId.isValid())
1544 return E_INVALIDARG;
1545
1546 HRESULT rc = i_updateNetIfList();
1547 if (FAILED(rc))
1548 {
1549 Log(("Failed to update host network interface list with rc=%Rhrc\n", rc));
1550 return rc;
1551 }
1552
1553 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1554
1555 ComObjPtr<HostNetworkInterface> found;
1556 for (HostNetworkInterfaceList::iterator it = m->llNetIfs.begin(); it != m->llNetIfs.end(); ++it)
1557 {
1558 Bstr g;
1559 (*it)->COMGETTER(Id)(g.asOutParam());
1560 if (Guid(g) == aId)
1561 found = *it;
1562 }
1563
1564 if (!found)
1565 return setError(E_INVALIDARG,
1566 HostNetworkInterface::tr("The host network interface with the given GUID could not be found"));
1567 return found.queryInterfaceTo(aNetworkInterface.asOutParam());
1568
1569#endif
1570}
1571
1572HRESULT Host::findHostNetworkInterfacesOfType(HostNetworkInterfaceType_T aType,
1573 std::vector<ComPtr<IHostNetworkInterface> > &aNetworkInterfaces)
1574{
1575#ifdef VBOX_WITH_HOSTNETIF_API
1576 HRESULT rc = i_updateNetIfList();
1577 if (FAILED(rc))
1578 {
1579 Log(("Failed to update host network interface list with rc=%Rhrc\n", rc));
1580 return rc;
1581 }
1582
1583 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1584
1585 HostNetworkInterfaceList resultList;
1586 for (HostNetworkInterfaceList::iterator it = m->llNetIfs.begin(); it != m->llNetIfs.end(); ++it)
1587 {
1588 HostNetworkInterfaceType_T t;
1589 HRESULT hr = (*it)->COMGETTER(InterfaceType)(&t);
1590 if (FAILED(hr))
1591 return hr;
1592
1593 if (t == aType)
1594 resultList.push_back(*it);
1595 }
1596 aNetworkInterfaces.resize(resultList.size());
1597 size_t i = 0;
1598 for (HostNetworkInterfaceList::iterator it = resultList.begin(); it != resultList.end(); ++it, ++i)
1599 {
1600 (*it).queryInterfaceTo(aNetworkInterfaces[i].asOutParam());
1601 }
1602
1603 return S_OK;
1604#else
1605 return E_NOTIMPL;
1606#endif
1607}
1608
1609HRESULT Host::findUSBDeviceByAddress(const com::Utf8Str &aName,
1610 ComPtr<IHostUSBDevice> &aDevice)
1611{
1612#ifdef VBOX_WITH_USB
1613
1614 aDevice = NULL;
1615 SafeIfaceArray<IHostUSBDevice> devsvec;
1616 HRESULT rc = COMGETTER(USBDevices)(ComSafeArrayAsOutParam(devsvec));
1617 if (FAILED(rc))
1618 return rc;
1619
1620 for (size_t i = 0; i < devsvec.size(); ++i)
1621 {
1622 Bstr address;
1623 rc = devsvec[i]->COMGETTER(Address)(address.asOutParam());
1624 if (FAILED(rc))
1625 return rc;
1626 if (address == aName)
1627 {
1628 return (ComPtr<IHostUSBDevice>(devsvec[i]).queryInterfaceTo(aDevice.asOutParam()));
1629 }
1630 }
1631
1632 return setErrorNoLog(VBOX_E_OBJECT_NOT_FOUND,
1633 tr("Could not find a USB device with address '%s'"),
1634 aName.c_str());
1635
1636#else /* !VBOX_WITH_USB */
1637 NOREF(aName);
1638 NOREF(aDevice);
1639 return E_NOTIMPL;
1640#endif /* !VBOX_WITH_USB */
1641}
1642HRESULT Host::findUSBDeviceById(const com::Guid &aId,
1643 ComPtr<IHostUSBDevice> &aDevice)
1644{
1645#ifdef VBOX_WITH_USB
1646 if (!aId.isValid())
1647 return E_INVALIDARG;
1648
1649 aDevice = NULL;
1650
1651 SafeIfaceArray<IHostUSBDevice> devsvec;
1652 HRESULT rc = COMGETTER(USBDevices)(ComSafeArrayAsOutParam(devsvec));
1653 if (FAILED(rc))
1654 return rc;
1655
1656 for (size_t i = 0; i < devsvec.size(); ++i)
1657 {
1658 Bstr id;
1659 rc = devsvec[i]->COMGETTER(Id)(id.asOutParam());
1660 if (FAILED(rc))
1661 return rc;
1662 if (Guid(id) == aId)
1663 {
1664 return (ComPtr<IHostUSBDevice>(devsvec[i]).queryInterfaceTo(aDevice.asOutParam()));
1665 }
1666 }
1667 return setErrorNoLog(VBOX_E_OBJECT_NOT_FOUND,
1668 tr("Could not find a USB device with uuid {%RTuuid}"),
1669 aId.raw());
1670
1671#else /* !VBOX_WITH_USB */
1672 NOREF(aId);
1673 NOREF(aDevice);
1674 return E_NOTIMPL;
1675#endif /* !VBOX_WITH_USB */
1676}
1677
1678HRESULT Host::generateMACAddress(com::Utf8Str &aAddress)
1679{
1680 // no locking required
1681 i_generateMACAddress(aAddress);
1682 return S_OK;
1683}
1684
1685/**
1686 * Returns a list of host video capture devices (webcams, etc).
1687 *
1688 * @returns COM status code
1689 * @param aVideoInputDevices Array of interface pointers to be filled.
1690 */
1691HRESULT Host::getVideoInputDevices(std::vector<ComPtr<IHostVideoInputDevice> > &aVideoInputDevices)
1692{
1693 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1694 HostVideoInputDeviceList list;
1695
1696 HRESULT rc = HostVideoInputDevice::queryHostDevices(m->pParent, &list);
1697 if (FAILED(rc))
1698 return rc;
1699
1700 aVideoInputDevices.resize(list.size());
1701 size_t i = 0;
1702 for (HostVideoInputDeviceList::const_iterator it = list.begin(); it != list.end(); ++it, ++i)
1703 (*it).queryInterfaceTo(aVideoInputDevices[i].asOutParam());
1704
1705 return S_OK;
1706}
1707
1708// public methods only for internal purposes
1709////////////////////////////////////////////////////////////////////////////////
1710
1711HRESULT Host::i_loadSettings(const settings::Host &data)
1712{
1713 HRESULT rc = S_OK;
1714#ifdef VBOX_WITH_USB
1715 AutoCaller autoCaller(this);
1716 if (FAILED(autoCaller.rc()))
1717 return autoCaller.rc();
1718
1719 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1720
1721 for (settings::USBDeviceFiltersList::const_iterator it = data.llUSBDeviceFilters.begin();
1722 it != data.llUSBDeviceFilters.end();
1723 ++it)
1724 {
1725 const settings::USBDeviceFilter &f = *it;
1726 ComObjPtr<HostUSBDeviceFilter> pFilter;
1727 pFilter.createObject();
1728 rc = pFilter->init(this, f);
1729 if (FAILED(rc))
1730 break;
1731
1732 m->llUSBDeviceFilters.push_back(pFilter);
1733 pFilter->mInList = true;
1734
1735 /* notify the proxy (only when the filter is active) */
1736 if (pFilter->i_getData().mActive)
1737 {
1738 HostUSBDeviceFilter *flt = pFilter; /* resolve ambiguity */
1739 flt->i_getId() = m->pUSBProxyService->insertFilter(&pFilter->i_getData().mUSBFilter);
1740 }
1741 }
1742#else
1743 NOREF(data);
1744#endif /* VBOX_WITH_USB */
1745 return rc;
1746}
1747
1748HRESULT Host::i_saveSettings(settings::Host &data)
1749{
1750#ifdef VBOX_WITH_USB
1751 AutoCaller autoCaller(this);
1752 if (FAILED(autoCaller.rc()))
1753 return autoCaller.rc();
1754
1755 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1756
1757 data.llUSBDeviceFilters.clear();
1758
1759 for (USBDeviceFilterList::const_iterator it = m->llUSBDeviceFilters.begin();
1760 it != m->llUSBDeviceFilters.end();
1761 ++it)
1762 {
1763 ComObjPtr<HostUSBDeviceFilter> pFilter = *it;
1764 settings::USBDeviceFilter f;
1765 pFilter->i_saveSettings(f);
1766 data.llUSBDeviceFilters.push_back(f);
1767 }
1768#else
1769 NOREF(data);
1770#endif /* VBOX_WITH_USB */
1771
1772 return S_OK;
1773}
1774
1775/**
1776 * Sets the given pointer to point to the static list of DVD or floppy
1777 * drives in the Host instance data, depending on the @a mediumType
1778 * parameter.
1779 *
1780 * This builds the list on the first call; it adds or removes host drives
1781 * that may have changed if fRefresh == true.
1782 *
1783 * The caller must hold the medium tree write lock before calling this.
1784 * To protect the list to which the caller's pointer points, the caller
1785 * must also hold that lock.
1786 *
1787 * @param mediumType Must be DeviceType_Floppy or DeviceType_DVD.
1788 * @param fRefresh Whether to refresh the host drives list even if this is not the first call.
1789 * @param pll Caller's pointer which gets set to the static list of host drives.
1790 * @param treeLock Reference to media tree lock, need to drop it temporarily.
1791 * @returns COM status code
1792 */
1793HRESULT Host::i_getDrives(DeviceType_T mediumType,
1794 bool fRefresh,
1795 MediaList *&pll,
1796 AutoWriteLock &treeLock)
1797{
1798 HRESULT rc = S_OK;
1799 Assert(m->pParent->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
1800
1801 MediaList llNew;
1802 MediaList *pllCached;
1803 bool *pfListBuilt = NULL;
1804
1805 switch (mediumType)
1806 {
1807 case DeviceType_DVD:
1808 if (!m->fDVDDrivesListBuilt || fRefresh)
1809 {
1810 rc = i_buildDVDDrivesList(llNew);
1811 if (FAILED(rc))
1812 return rc;
1813 pfListBuilt = &m->fDVDDrivesListBuilt;
1814 }
1815 pllCached = &m->llDVDDrives;
1816 break;
1817
1818 case DeviceType_Floppy:
1819 if (!m->fFloppyDrivesListBuilt || fRefresh)
1820 {
1821 rc = i_buildFloppyDrivesList(llNew);
1822 if (FAILED(rc))
1823 return rc;
1824 pfListBuilt = &m->fFloppyDrivesListBuilt;
1825 }
1826 pllCached = &m->llFloppyDrives;
1827 break;
1828
1829 default:
1830 return E_INVALIDARG;
1831 }
1832
1833 if (pfListBuilt)
1834 {
1835 // a list was built in llNew above:
1836 if (!*pfListBuilt)
1837 {
1838 // this was the first call (instance bool is still false): then just copy the whole list and return
1839 *pllCached = llNew;
1840 // and mark the instance data as "built"
1841 *pfListBuilt = true;
1842 }
1843 else
1844 {
1845 // list was built, and this was a subsequent call: then compare the old and the new lists
1846
1847 // remove drives from the cached list which are no longer present
1848 for (MediaList::iterator itCached = pllCached->begin();
1849 itCached != pllCached->end();
1850 /*nothing */)
1851 {
1852 Medium *pCached = *itCached;
1853 const Utf8Str strLocationCached = pCached->i_getLocationFull();
1854 bool fFound = false;
1855 for (MediaList::iterator itNew = llNew.begin();
1856 itNew != llNew.end();
1857 ++itNew)
1858 {
1859 Medium *pNew = *itNew;
1860 const Utf8Str strLocationNew = pNew->i_getLocationFull();
1861 if (strLocationNew == strLocationCached)
1862 {
1863 fFound = true;
1864 break;
1865 }
1866 }
1867 if (!fFound)
1868 {
1869 pCached->uninit();
1870 itCached = pllCached->erase(itCached);
1871 }
1872 else
1873 ++itCached;
1874 }
1875
1876 // add drives to the cached list that are not on there yet
1877 for (MediaList::iterator itNew = llNew.begin();
1878 itNew != llNew.end();
1879 ++itNew)
1880 {
1881 Medium *pNew = *itNew;
1882 const Utf8Str strLocationNew = pNew->i_getLocationFull();
1883 bool fFound = false;
1884 for (MediaList::iterator itCached = pllCached->begin();
1885 itCached != pllCached->end();
1886 ++itCached)
1887 {
1888 Medium *pCached = *itCached;
1889 const Utf8Str strLocationCached = pCached->i_getLocationFull();
1890 if (strLocationNew == strLocationCached)
1891 {
1892 fFound = true;
1893 break;
1894 }
1895 }
1896
1897 if (!fFound)
1898 pllCached->push_back(pNew);
1899 }
1900 }
1901 }
1902
1903 // return cached list to caller
1904 pll = pllCached;
1905
1906 // Make sure the media tree lock is released before llNew is cleared,
1907 // as this usually triggers calls to uninit().
1908 treeLock.release();
1909
1910 llNew.clear();
1911
1912 treeLock.acquire();
1913
1914 return rc;
1915}
1916
1917/**
1918 * Goes through the list of host drives that would be returned by getDrives()
1919 * and looks for a host drive with the given UUID. If found, it sets pMedium
1920 * to that drive; otherwise returns VBOX_E_OBJECT_NOT_FOUND.
1921 *
1922 * @param mediumType Must be DeviceType_DVD or DeviceType_Floppy.
1923 * @param uuid Medium UUID of host drive to look for.
1924 * @param fRefresh Whether to refresh the host drives list (see getDrives())
1925 * @param pMedium Medium object, if found...
1926 * @return VBOX_E_OBJECT_NOT_FOUND if not found, or S_OK if found, or errors from getDrives().
1927 */
1928HRESULT Host::i_findHostDriveById(DeviceType_T mediumType,
1929 const Guid &uuid,
1930 bool fRefresh,
1931 ComObjPtr<Medium> &pMedium)
1932{
1933 MediaList *pllMedia;
1934
1935 AutoWriteLock treeLock(m->pParent->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1936 HRESULT rc = i_getDrives(mediumType, fRefresh, pllMedia, treeLock);
1937 if (SUCCEEDED(rc))
1938 {
1939 for (MediaList::iterator it = pllMedia->begin();
1940 it != pllMedia->end();
1941 ++it)
1942 {
1943 Medium *pThis = *it;
1944 AutoCaller mediumCaller(pThis);
1945 AutoReadLock mediumLock(pThis COMMA_LOCKVAL_SRC_POS);
1946 if (pThis->i_getId() == uuid)
1947 {
1948 pMedium = pThis;
1949 return S_OK;
1950 }
1951 }
1952 }
1953
1954 return VBOX_E_OBJECT_NOT_FOUND;
1955}
1956
1957/**
1958 * Goes through the list of host drives that would be returned by getDrives()
1959 * and looks for a host drive with the given name. If found, it sets pMedium
1960 * to that drive; otherwise returns VBOX_E_OBJECT_NOT_FOUND.
1961 *
1962 * @param mediumType Must be DeviceType_DVD or DeviceType_Floppy.
1963 * @param strLocationFull Name (path) of host drive to look for.
1964 * @param fRefresh Whether to refresh the host drives list (see getDrives())
1965 * @param pMedium Medium object, if found
1966 * @return VBOX_E_OBJECT_NOT_FOUND if not found, or S_OK if found, or errors from getDrives().
1967 */
1968HRESULT Host::i_findHostDriveByName(DeviceType_T mediumType,
1969 const Utf8Str &strLocationFull,
1970 bool fRefresh,
1971 ComObjPtr<Medium> &pMedium)
1972{
1973 MediaList *pllMedia;
1974
1975 AutoWriteLock treeLock(m->pParent->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1976 HRESULT rc = i_getDrives(mediumType, fRefresh, pllMedia, treeLock);
1977 if (SUCCEEDED(rc))
1978 {
1979 for (MediaList::iterator it = pllMedia->begin();
1980 it != pllMedia->end();
1981 ++it)
1982 {
1983 Medium *pThis = *it;
1984 AutoCaller mediumCaller(pThis);
1985 AutoReadLock mediumLock(pThis COMMA_LOCKVAL_SRC_POS);
1986 if (pThis->i_getLocationFull() == strLocationFull)
1987 {
1988 pMedium = pThis;
1989 return S_OK;
1990 }
1991 }
1992 }
1993
1994 return VBOX_E_OBJECT_NOT_FOUND;
1995}
1996
1997/**
1998 * Goes through the list of host drives that would be returned by getDrives()
1999 * and looks for a host drive with the given name, location or ID. If found,
2000 * it sets pMedium to that drive; otherwise returns VBOX_E_OBJECT_NOT_FOUND.
2001 *
2002 * @param mediumType Must be DeviceType_DVD or DeviceType_Floppy.
2003 * @param strNameOrId Name or full location or UUID of host drive to look for.
2004 * @param pMedium Medium object, if found...
2005 * @return VBOX_E_OBJECT_NOT_FOUND if not found, or S_OK if found, or errors from getDrives().
2006 */
2007HRESULT Host::i_findHostDriveByNameOrId(DeviceType_T mediumType,
2008 const Utf8Str &strNameOrId,
2009 ComObjPtr<Medium> &pMedium)
2010{
2011 AutoWriteLock wlock(m->pParent->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
2012
2013 Guid uuid(strNameOrId);
2014 if (uuid.isValid() && !uuid.isZero())
2015 return i_findHostDriveById(mediumType, uuid, true /* fRefresh */, pMedium);
2016
2017 // string is not a syntactically valid UUID: try a name then
2018 return i_findHostDriveByName(mediumType, strNameOrId, true /* fRefresh */, pMedium);
2019}
2020
2021/**
2022 * Called from getDrives() to build the DVD drives list.
2023 * @param pll
2024 * @return
2025 */
2026HRESULT Host::i_buildDVDDrivesList(MediaList &list)
2027{
2028 HRESULT rc = S_OK;
2029
2030 Assert(m->pParent->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
2031
2032 try
2033 {
2034#if defined(RT_OS_WINDOWS)
2035 int sz = GetLogicalDriveStrings(0, NULL);
2036 TCHAR *hostDrives = new TCHAR[sz+1];
2037 GetLogicalDriveStrings(sz, hostDrives);
2038 wchar_t driveName[3] = { '?', ':', '\0' };
2039 TCHAR *p = hostDrives;
2040 do
2041 {
2042 if (GetDriveType(p) == DRIVE_CDROM)
2043 {
2044 driveName[0] = *p;
2045 ComObjPtr<Medium> hostDVDDriveObj;
2046 hostDVDDriveObj.createObject();
2047 hostDVDDriveObj->init(m->pParent, DeviceType_DVD, Bstr(driveName));
2048 list.push_back(hostDVDDriveObj);
2049 }
2050 p += _tcslen(p) + 1;
2051 }
2052 while (*p);
2053 delete[] hostDrives;
2054
2055#elif defined(RT_OS_SOLARIS)
2056# ifdef VBOX_USE_LIBHAL
2057 if (!i_getDVDInfoFromHal(list))
2058# endif
2059 {
2060 i_getDVDInfoFromDevTree(list);
2061 }
2062
2063#elif defined(RT_OS_LINUX) || defined(RT_OS_FREEBSD)
2064 if (RT_SUCCESS(m->hostDrives.updateDVDs()))
2065 for (DriveInfoList::const_iterator it = m->hostDrives.DVDBegin();
2066 SUCCEEDED(rc) && it != m->hostDrives.DVDEnd(); ++it)
2067 {
2068 ComObjPtr<Medium> hostDVDDriveObj;
2069 Utf8Str location(it->mDevice);
2070 Utf8Str description(it->mDescription);
2071 if (SUCCEEDED(rc))
2072 rc = hostDVDDriveObj.createObject();
2073 if (SUCCEEDED(rc))
2074 rc = hostDVDDriveObj->init(m->pParent, DeviceType_DVD, location, description);
2075 if (SUCCEEDED(rc))
2076 list.push_back(hostDVDDriveObj);
2077 }
2078#elif defined(RT_OS_DARWIN)
2079 PDARWINDVD cur = DarwinGetDVDDrives();
2080 while (cur)
2081 {
2082 ComObjPtr<Medium> hostDVDDriveObj;
2083 hostDVDDriveObj.createObject();
2084 hostDVDDriveObj->init(m->pParent, DeviceType_DVD, Bstr(cur->szName));
2085 list.push_back(hostDVDDriveObj);
2086
2087 /* next */
2088 void *freeMe = cur;
2089 cur = cur->pNext;
2090 RTMemFree(freeMe);
2091 }
2092#else
2093 /* PORTME */
2094#endif
2095 }
2096 catch(std::bad_alloc &)
2097 {
2098 rc = E_OUTOFMEMORY;
2099 }
2100 return rc;
2101}
2102
2103/**
2104 * Called from getDrives() to build the floppy drives list.
2105 * @param list
2106 * @return
2107 */
2108HRESULT Host::i_buildFloppyDrivesList(MediaList &list)
2109{
2110 HRESULT rc = S_OK;
2111
2112 Assert(m->pParent->i_getMediaTreeLockHandle().isWriteLockOnCurrentThread());
2113
2114 try
2115 {
2116#ifdef RT_OS_WINDOWS
2117 int sz = GetLogicalDriveStrings(0, NULL);
2118 TCHAR *hostDrives = new TCHAR[sz+1];
2119 GetLogicalDriveStrings(sz, hostDrives);
2120 wchar_t driveName[3] = { '?', ':', '\0' };
2121 TCHAR *p = hostDrives;
2122 do
2123 {
2124 if (GetDriveType(p) == DRIVE_REMOVABLE)
2125 {
2126 driveName[0] = *p;
2127 ComObjPtr<Medium> hostFloppyDriveObj;
2128 hostFloppyDriveObj.createObject();
2129 hostFloppyDriveObj->init(m->pParent, DeviceType_Floppy, Bstr(driveName));
2130 list.push_back(hostFloppyDriveObj);
2131 }
2132 p += _tcslen(p) + 1;
2133 }
2134 while (*p);
2135 delete[] hostDrives;
2136#elif defined(RT_OS_LINUX)
2137 if (RT_SUCCESS(m->hostDrives.updateFloppies()))
2138 for (DriveInfoList::const_iterator it = m->hostDrives.FloppyBegin();
2139 SUCCEEDED(rc) && it != m->hostDrives.FloppyEnd(); ++it)
2140 {
2141 ComObjPtr<Medium> hostFloppyDriveObj;
2142 Utf8Str location(it->mDevice);
2143 Utf8Str description(it->mDescription);
2144 if (SUCCEEDED(rc))
2145 rc = hostFloppyDriveObj.createObject();
2146 if (SUCCEEDED(rc))
2147 rc = hostFloppyDriveObj->init(m->pParent, DeviceType_Floppy, location, description);
2148 if (SUCCEEDED(rc))
2149 list.push_back(hostFloppyDriveObj);
2150 }
2151#else
2152 NOREF(list);
2153 /* PORTME */
2154#endif
2155 }
2156 catch(std::bad_alloc &)
2157 {
2158 rc = E_OUTOFMEMORY;
2159 }
2160
2161 return rc;
2162}
2163
2164#ifdef VBOX_WITH_USB
2165USBProxyService* Host::i_usbProxyService()
2166{
2167 return m->pUSBProxyService;
2168}
2169
2170HRESULT Host::i_addChild(HostUSBDeviceFilter *pChild)
2171{
2172 AutoCaller autoCaller(this);
2173 if (FAILED(autoCaller.rc()))
2174 return autoCaller.rc();
2175
2176 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2177
2178 m->llChildren.push_back(pChild);
2179
2180 return S_OK;
2181}
2182
2183HRESULT Host::i_removeChild(HostUSBDeviceFilter *pChild)
2184{
2185 AutoCaller autoCaller(this);
2186 if (FAILED(autoCaller.rc()))
2187 return autoCaller.rc();
2188
2189 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2190
2191 for (USBDeviceFilterList::iterator it = m->llChildren.begin();
2192 it != m->llChildren.end();
2193 ++it)
2194 {
2195 if (*it == pChild)
2196 {
2197 m->llChildren.erase(it);
2198 break;
2199 }
2200 }
2201
2202 return S_OK;
2203}
2204
2205VirtualBox* Host::i_parent()
2206{
2207 return m->pParent;
2208}
2209
2210/**
2211 * Called by setter methods of all USB device filters.
2212 */
2213HRESULT Host::i_onUSBDeviceFilterChange(HostUSBDeviceFilter *aFilter,
2214 BOOL aActiveChanged /* = FALSE */)
2215{
2216 AutoCaller autoCaller(this);
2217 if (FAILED(autoCaller.rc()))
2218 return autoCaller.rc();
2219
2220 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2221
2222 if (aFilter->mInList)
2223 {
2224 if (aActiveChanged)
2225 {
2226 // insert/remove the filter from the proxy
2227 if (aFilter->i_getData().mActive)
2228 {
2229 ComAssertRet(aFilter->i_getId() == NULL, E_FAIL);
2230 aFilter->i_getId() = m->pUSBProxyService->insertFilter(&aFilter->i_getData().mUSBFilter);
2231 }
2232 else
2233 {
2234 ComAssertRet(aFilter->i_getId() != NULL, E_FAIL);
2235 m->pUSBProxyService->removeFilter(aFilter->i_getId());
2236 aFilter->i_getId() = NULL;
2237 }
2238 }
2239 else
2240 {
2241 if (aFilter->i_getData().mActive)
2242 {
2243 // update the filter in the proxy
2244 ComAssertRet(aFilter->i_getId() != NULL, E_FAIL);
2245 m->pUSBProxyService->removeFilter(aFilter->i_getId());
2246 aFilter->i_getId() = m->pUSBProxyService->insertFilter(&aFilter->i_getData().mUSBFilter);
2247 }
2248 }
2249
2250 // save the global settings... yeah, on every single filter property change
2251 // for that we should hold only the VirtualBox lock
2252 alock.release();
2253 AutoWriteLock vboxLock(m->pParent COMMA_LOCKVAL_SRC_POS);
2254 return m->pParent->i_saveSettings();
2255 }
2256
2257 return S_OK;
2258}
2259
2260
2261/**
2262 * Interface for obtaining a copy of the USBDeviceFilterList,
2263 * used by the USBProxyService.
2264 *
2265 * @param aGlobalFilters Where to put the global filter list copy.
2266 * @param aMachines Where to put the machine vector.
2267 */
2268void Host::i_getUSBFilters(Host::USBDeviceFilterList *aGlobalFilters)
2269{
2270 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
2271
2272 *aGlobalFilters = m->llUSBDeviceFilters;
2273}
2274
2275#endif /* VBOX_WITH_USB */
2276
2277// private methods
2278////////////////////////////////////////////////////////////////////////////////
2279
2280#if defined(RT_OS_SOLARIS) && defined(VBOX_USE_LIBHAL)
2281
2282/**
2283 * Helper function to get the slice number from a device path
2284 *
2285 * @param pszDevLinkPath Pointer to a device path (/dev/(r)dsk/c7d1t0d0s3 etc.)
2286 * @returns Pointer to the slice portion of the given path.
2287 */
2288static char *solarisGetSliceFromPath(const char *pszDevLinkPath)
2289{
2290 char *pszFound = NULL;
2291 char *pszSlice = strrchr(pszDevLinkPath, 's');
2292 char *pszDisk = strrchr(pszDevLinkPath, 'd');
2293 if (pszSlice && pszSlice > pszDisk)
2294 pszFound = pszSlice;
2295 else
2296 pszFound = pszDisk;
2297
2298 if (pszFound && RT_C_IS_DIGIT(pszFound[1]))
2299 return pszFound;
2300
2301 return NULL;
2302}
2303
2304/**
2305 * Walk device links and returns an allocated path for the first one in the snapshot.
2306 *
2307 * @param DevLink Handle to the device link being walked.
2308 * @param pvArg Opaque data containing the pointer to the path.
2309 * @returns Pointer to an allocated device path string.
2310 */
2311static int solarisWalkDevLink(di_devlink_t DevLink, void *pvArg)
2312{
2313 char **ppszPath = (char **)pvArg;
2314 *ppszPath = strdup(di_devlink_path(DevLink));
2315 return DI_WALK_TERMINATE;
2316}
2317
2318/**
2319 * Walk all devices in the system and enumerate CD/DVD drives.
2320 * @param Node Handle to the current node.
2321 * @param pvArg Opaque data (holds list pointer).
2322 * @returns Solaris specific code whether to continue walking or not.
2323 */
2324static int solarisWalkDeviceNodeForDVD(di_node_t Node, void *pvArg)
2325{
2326 PSOLARISDVD *ppDrives = (PSOLARISDVD *)pvArg;
2327
2328 /*
2329 * Check for "removable-media" or "hotpluggable" instead of "SCSI" so that we also include USB CD-ROMs.
2330 * As unfortunately the Solaris drivers only export these common properties.
2331 */
2332 int *pInt = NULL;
2333 if ( di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "removable-media", &pInt) >= 0
2334 || di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "hotpluggable", &pInt) >= 0)
2335 {
2336 if (di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "inquiry-device-type", &pInt) > 0
2337 && ( *pInt == DTYPE_RODIRECT /* CDROM */
2338 || *pInt == DTYPE_OPTICAL)) /* Optical Drive */
2339 {
2340 char *pszProduct = NULL;
2341 if (di_prop_lookup_strings(DDI_DEV_T_ANY, Node, "inquiry-product-id", &pszProduct) > 0)
2342 {
2343 char *pszVendor = NULL;
2344 if (di_prop_lookup_strings(DDI_DEV_T_ANY, Node, "inquiry-vendor-id", &pszVendor) > 0)
2345 {
2346 /*
2347 * Found a DVD drive, we need to scan the minor nodes to find the correct
2348 * slice that represents the whole drive. "s2" is always the whole drive for CD/DVDs.
2349 */
2350 int Major = di_driver_major(Node);
2351 di_minor_t Minor = DI_MINOR_NIL;
2352 di_devlink_handle_t DevLink = di_devlink_init(NULL /* name */, 0 /* flags */);
2353 if (DevLink)
2354 {
2355 while ((Minor = di_minor_next(Node, Minor)) != DI_MINOR_NIL)
2356 {
2357 dev_t Dev = di_minor_devt(Minor);
2358 if ( Major != (int)major(Dev)
2359 || di_minor_spectype(Minor) == S_IFBLK
2360 || di_minor_type(Minor) != DDM_MINOR)
2361 {
2362 continue;
2363 }
2364
2365 char *pszMinorPath = di_devfs_minor_path(Minor);
2366 if (!pszMinorPath)
2367 continue;
2368
2369 char *pszDevLinkPath = NULL;
2370 di_devlink_walk(DevLink, NULL, pszMinorPath, DI_PRIMARY_LINK, &pszDevLinkPath, solarisWalkDevLink);
2371 di_devfs_path_free(pszMinorPath);
2372
2373 if (pszDevLinkPath)
2374 {
2375 char *pszSlice = solarisGetSliceFromPath(pszDevLinkPath);
2376 if ( pszSlice && !strcmp(pszSlice, "s2")
2377 && !strncmp(pszDevLinkPath, RT_STR_TUPLE("/dev/rdsk"))) /* We want only raw disks */
2378 {
2379 /*
2380 * We've got a fully qualified DVD drive. Add it to the list.
2381 */
2382 PSOLARISDVD pDrive = (PSOLARISDVD)RTMemAllocZ(sizeof(SOLARISDVD));
2383 if (RT_LIKELY(pDrive))
2384 {
2385 RTStrPrintf(pDrive->szDescription, sizeof(pDrive->szDescription),
2386 "%s %s", pszVendor, pszProduct);
2387 RTStrCopy(pDrive->szRawDiskPath, sizeof(pDrive->szRawDiskPath), pszDevLinkPath);
2388 if (*ppDrives)
2389 pDrive->pNext = *ppDrives;
2390 *ppDrives = pDrive;
2391
2392 /* We're not interested in any of the other slices, stop minor nodes traversal. */
2393 free(pszDevLinkPath);
2394 break;
2395 }
2396 }
2397 free(pszDevLinkPath);
2398 }
2399 }
2400 di_devlink_fini(&DevLink);
2401 }
2402 }
2403 }
2404 }
2405 }
2406 return DI_WALK_CONTINUE;
2407}
2408
2409/**
2410 * Solaris specific function to enumerate CD/DVD drives via the device tree.
2411 * Works on Solaris 10 as well as OpenSolaris without depending on libhal.
2412 */
2413void Host::i_getDVDInfoFromDevTree(std::list<ComObjPtr<Medium> > &list)
2414{
2415 PSOLARISDVD pDrives = NULL;
2416 di_node_t RootNode = di_init("/", DINFOCPYALL);
2417 if (RootNode != DI_NODE_NIL)
2418 di_walk_node(RootNode, DI_WALK_CLDFIRST, &pDrives, solarisWalkDeviceNodeForDVD);
2419
2420 di_fini(RootNode);
2421
2422 while (pDrives)
2423 {
2424 ComObjPtr<Medium> hostDVDDriveObj;
2425 hostDVDDriveObj.createObject();
2426 hostDVDDriveObj->init(m->pParent, DeviceType_DVD, Bstr(pDrives->szRawDiskPath), Bstr(pDrives->szDescription));
2427 list.push_back(hostDVDDriveObj);
2428
2429 void *pvDrive = pDrives;
2430 pDrives = pDrives->pNext;
2431 RTMemFree(pvDrive);
2432 }
2433}
2434
2435/* Solaris hosts, loading libhal at runtime */
2436
2437/**
2438 * Helper function to query the hal subsystem for information about DVD drives attached to the
2439 * system.
2440 *
2441 * @returns true if information was successfully obtained, false otherwise
2442 * @retval list drives found will be attached to this list
2443 */
2444bool Host::i_getDVDInfoFromHal(std::list<ComObjPtr<Medium> > &list)
2445{
2446 bool halSuccess = false;
2447 DBusError dbusError;
2448 if (!gLibHalCheckPresence())
2449 return false;
2450 gDBusErrorInit(&dbusError);
2451 DBusConnection *dbusConnection = gDBusBusGet(DBUS_BUS_SYSTEM, &dbusError);
2452 if (dbusConnection != 0)
2453 {
2454 LibHalContext *halContext = gLibHalCtxNew();
2455 if (halContext != 0)
2456 {
2457 if (gLibHalCtxSetDBusConnection(halContext, dbusConnection))
2458 {
2459 if (gLibHalCtxInit(halContext, &dbusError))
2460 {
2461 int numDevices;
2462 char **halDevices = gLibHalFindDeviceStringMatch(halContext,
2463 "storage.drive_type", "cdrom",
2464 &numDevices, &dbusError);
2465 if (halDevices != 0)
2466 {
2467 /* Hal is installed and working, so if no devices are reported, assume
2468 that there are none. */
2469 halSuccess = true;
2470 for (int i = 0; i < numDevices; i++)
2471 {
2472 char *devNode = gLibHalDeviceGetPropertyString(halContext,
2473 halDevices[i], "block.device", &dbusError);
2474#ifdef RT_OS_SOLARIS
2475 /* The CD/DVD ioctls work only for raw device nodes. */
2476 char *tmp = getfullrawname(devNode);
2477 gLibHalFreeString(devNode);
2478 devNode = tmp;
2479#endif
2480
2481 if (devNode != 0)
2482 {
2483// if (validateDevice(devNode, true))
2484// {
2485 Utf8Str description;
2486 char *vendor, *product;
2487 /* We do not check the error here, as this field may
2488 not even exist. */
2489 vendor = gLibHalDeviceGetPropertyString(halContext,
2490 halDevices[i], "info.vendor", 0);
2491 product = gLibHalDeviceGetPropertyString(halContext,
2492 halDevices[i], "info.product", &dbusError);
2493 if ((product != 0 && product[0] != 0))
2494 {
2495 if ((vendor != 0) && (vendor[0] != 0))
2496 {
2497 description = Utf8StrFmt("%s %s",
2498 vendor, product);
2499 }
2500 else
2501 {
2502 description = product;
2503 }
2504 ComObjPtr<Medium> hostDVDDriveObj;
2505 hostDVDDriveObj.createObject();
2506 hostDVDDriveObj->init(m->pParent, DeviceType_DVD,
2507 Bstr(devNode), Bstr(description));
2508 list.push_back(hostDVDDriveObj);
2509 }
2510 else
2511 {
2512 if (product == 0)
2513 {
2514 LogRel(("Host::COMGETTER(DVDDrives): failed to get property \"info.product\" for device %s. dbus error: %s (%s)\n",
2515 halDevices[i], dbusError.name, dbusError.message));
2516 gDBusErrorFree(&dbusError);
2517 }
2518 ComObjPtr<Medium> hostDVDDriveObj;
2519 hostDVDDriveObj.createObject();
2520 hostDVDDriveObj->init(m->pParent, DeviceType_DVD,
2521 Bstr(devNode));
2522 list.push_back(hostDVDDriveObj);
2523 }
2524 if (vendor != 0)
2525 {
2526 gLibHalFreeString(vendor);
2527 }
2528 if (product != 0)
2529 {
2530 gLibHalFreeString(product);
2531 }
2532// }
2533// else
2534// {
2535// LogRel(("Host::COMGETTER(DVDDrives): failed to validate the block device %s as a DVD drive\n"));
2536// }
2537#ifndef RT_OS_SOLARIS
2538 gLibHalFreeString(devNode);
2539#else
2540 free(devNode);
2541#endif
2542 }
2543 else
2544 {
2545 LogRel(("Host::COMGETTER(DVDDrives): failed to get property \"block.device\" for device %s. dbus error: %s (%s)\n",
2546 halDevices[i], dbusError.name, dbusError.message));
2547 gDBusErrorFree(&dbusError);
2548 }
2549 }
2550 gLibHalFreeStringArray(halDevices);
2551 }
2552 else
2553 {
2554 LogRel(("Host::COMGETTER(DVDDrives): failed to get devices with capability \"storage.cdrom\". dbus error: %s (%s)\n", dbusError.name, dbusError.message));
2555 gDBusErrorFree(&dbusError);
2556 }
2557 if (!gLibHalCtxShutdown(halContext, &dbusError)) /* what now? */
2558 {
2559 LogRel(("Host::COMGETTER(DVDDrives): failed to shutdown the libhal context. dbus error: %s (%s)\n",
2560 dbusError.name, dbusError.message));
2561 gDBusErrorFree(&dbusError);
2562 }
2563 }
2564 else
2565 {
2566 LogRel(("Host::COMGETTER(DVDDrives): failed to initialise libhal context. dbus error: %s (%s)\n",
2567 dbusError.name, dbusError.message));
2568 gDBusErrorFree(&dbusError);
2569 }
2570 gLibHalCtxFree(halContext);
2571 }
2572 else
2573 {
2574 LogRel(("Host::COMGETTER(DVDDrives): failed to set libhal connection to dbus.\n"));
2575 }
2576 }
2577 else
2578 {
2579 LogRel(("Host::COMGETTER(DVDDrives): failed to get a libhal context - out of memory?\n"));
2580 }
2581 gDBusConnectionUnref(dbusConnection);
2582 }
2583 else
2584 {
2585 LogRel(("Host::COMGETTER(DVDDrives): failed to connect to dbus. dbus error: %s (%s)\n",
2586 dbusError.name, dbusError.message));
2587 gDBusErrorFree(&dbusError);
2588 }
2589 return halSuccess;
2590}
2591
2592
2593/**
2594 * Helper function to query the hal subsystem for information about floppy drives attached to the
2595 * system.
2596 *
2597 * @returns true if information was successfully obtained, false otherwise
2598 * @retval list drives found will be attached to this list
2599 */
2600bool Host::i_getFloppyInfoFromHal(std::list< ComObjPtr<Medium> > &list)
2601{
2602 bool halSuccess = false;
2603 DBusError dbusError;
2604 if (!gLibHalCheckPresence())
2605 return false;
2606 gDBusErrorInit(&dbusError);
2607 DBusConnection *dbusConnection = gDBusBusGet(DBUS_BUS_SYSTEM, &dbusError);
2608 if (dbusConnection != 0)
2609 {
2610 LibHalContext *halContext = gLibHalCtxNew();
2611 if (halContext != 0)
2612 {
2613 if (gLibHalCtxSetDBusConnection(halContext, dbusConnection))
2614 {
2615 if (gLibHalCtxInit(halContext, &dbusError))
2616 {
2617 int numDevices;
2618 char **halDevices = gLibHalFindDeviceStringMatch(halContext,
2619 "storage.drive_type", "floppy",
2620 &numDevices, &dbusError);
2621 if (halDevices != 0)
2622 {
2623 /* Hal is installed and working, so if no devices are reported, assume
2624 that there are none. */
2625 halSuccess = true;
2626 for (int i = 0; i < numDevices; i++)
2627 {
2628 char *driveType = gLibHalDeviceGetPropertyString(halContext,
2629 halDevices[i], "storage.drive_type", 0);
2630 if (driveType != 0)
2631 {
2632 if (strcmp(driveType, "floppy") != 0)
2633 {
2634 gLibHalFreeString(driveType);
2635 continue;
2636 }
2637 gLibHalFreeString(driveType);
2638 }
2639 else
2640 {
2641 /* An error occurred. The attribute "storage.drive_type"
2642 probably didn't exist. */
2643 continue;
2644 }
2645 char *devNode = gLibHalDeviceGetPropertyString(halContext,
2646 halDevices[i], "block.device", &dbusError);
2647 if (devNode != 0)
2648 {
2649// if (validateDevice(devNode, false))
2650// {
2651 Utf8Str description;
2652 char *vendor, *product;
2653 /* We do not check the error here, as this field may
2654 not even exist. */
2655 vendor = gLibHalDeviceGetPropertyString(halContext,
2656 halDevices[i], "info.vendor", 0);
2657 product = gLibHalDeviceGetPropertyString(halContext,
2658 halDevices[i], "info.product", &dbusError);
2659 if ((product != 0) && (product[0] != 0))
2660 {
2661 if ((vendor != 0) && (vendor[0] != 0))
2662 {
2663 description = Utf8StrFmt("%s %s",
2664 vendor, product);
2665 }
2666 else
2667 {
2668 description = product;
2669 }
2670 ComObjPtr<Medium> hostFloppyDrive;
2671 hostFloppyDrive.createObject();
2672 hostFloppyDrive->init(m->pParent, DeviceType_DVD,
2673 Bstr(devNode), Bstr(description));
2674 list.push_back(hostFloppyDrive);
2675 }
2676 else
2677 {
2678 if (product == 0)
2679 {
2680 LogRel(("Host::COMGETTER(FloppyDrives): failed to get property \"info.product\" for device %s. dbus error: %s (%s)\n",
2681 halDevices[i], dbusError.name, dbusError.message));
2682 gDBusErrorFree(&dbusError);
2683 }
2684 ComObjPtr<Medium> hostFloppyDrive;
2685 hostFloppyDrive.createObject();
2686 hostFloppyDrive->init(m->pParent, DeviceType_DVD,
2687 Bstr(devNode));
2688 list.push_back(hostFloppyDrive);
2689 }
2690 if (vendor != 0)
2691 {
2692 gLibHalFreeString(vendor);
2693 }
2694 if (product != 0)
2695 {
2696 gLibHalFreeString(product);
2697 }
2698// }
2699// else
2700// {
2701// LogRel(("Host::COMGETTER(FloppyDrives): failed to validate the block device %s as a floppy drive\n"));
2702// }
2703 gLibHalFreeString(devNode);
2704 }
2705 else
2706 {
2707 LogRel(("Host::COMGETTER(FloppyDrives): failed to get property \"block.device\" for device %s. dbus error: %s (%s)\n",
2708 halDevices[i], dbusError.name, dbusError.message));
2709 gDBusErrorFree(&dbusError);
2710 }
2711 }
2712 gLibHalFreeStringArray(halDevices);
2713 }
2714 else
2715 {
2716 LogRel(("Host::COMGETTER(FloppyDrives): failed to get devices with capability \"storage.cdrom\". dbus error: %s (%s)\n", dbusError.name, dbusError.message));
2717 gDBusErrorFree(&dbusError);
2718 }
2719 if (!gLibHalCtxShutdown(halContext, &dbusError)) /* what now? */
2720 {
2721 LogRel(("Host::COMGETTER(FloppyDrives): failed to shutdown the libhal context. dbus error: %s (%s)\n",
2722 dbusError.name, dbusError.message));
2723 gDBusErrorFree(&dbusError);
2724 }
2725 }
2726 else
2727 {
2728 LogRel(("Host::COMGETTER(FloppyDrives): failed to initialise libhal context. dbus error: %s (%s)\n",
2729 dbusError.name, dbusError.message));
2730 gDBusErrorFree(&dbusError);
2731 }
2732 gLibHalCtxFree(halContext);
2733 }
2734 else
2735 {
2736 LogRel(("Host::COMGETTER(FloppyDrives): failed to set libhal connection to dbus.\n"));
2737 }
2738 }
2739 else
2740 {
2741 LogRel(("Host::COMGETTER(FloppyDrives): failed to get a libhal context - out of memory?\n"));
2742 }
2743 gDBusConnectionUnref(dbusConnection);
2744 }
2745 else
2746 {
2747 LogRel(("Host::COMGETTER(FloppyDrives): failed to connect to dbus. dbus error: %s (%s)\n",
2748 dbusError.name, dbusError.message));
2749 gDBusErrorFree(&dbusError);
2750 }
2751 return halSuccess;
2752}
2753#endif /* RT_OS_SOLARIS and VBOX_USE_HAL */
2754
2755/** @todo get rid of dead code below - RT_OS_SOLARIS and RT_OS_LINUX are never both set */
2756#if defined(RT_OS_SOLARIS)
2757
2758/**
2759 * Helper function to parse the given mount file and add found entries
2760 */
2761void Host::i_parseMountTable(char *mountTable, std::list< ComObjPtr<Medium> > &list)
2762{
2763#ifdef RT_OS_LINUX
2764 FILE *mtab = setmntent(mountTable, "r");
2765 if (mtab)
2766 {
2767 struct mntent *mntent;
2768 char *mnt_type;
2769 char *mnt_dev;
2770 char *tmp;
2771 while ((mntent = getmntent(mtab)))
2772 {
2773 mnt_type = (char*)malloc(strlen(mntent->mnt_type) + 1);
2774 mnt_dev = (char*)malloc(strlen(mntent->mnt_fsname) + 1);
2775 strcpy(mnt_type, mntent->mnt_type);
2776 strcpy(mnt_dev, mntent->mnt_fsname);
2777 // supermount fs case
2778 if (strcmp(mnt_type, "supermount") == 0)
2779 {
2780 tmp = strstr(mntent->mnt_opts, "fs=");
2781 if (tmp)
2782 {
2783 free(mnt_type);
2784 mnt_type = strdup(tmp + strlen("fs="));
2785 if (mnt_type)
2786 {
2787 tmp = strchr(mnt_type, ',');
2788 if (tmp)
2789 *tmp = '\0';
2790 }
2791 }
2792 tmp = strstr(mntent->mnt_opts, "dev=");
2793 if (tmp)
2794 {
2795 free(mnt_dev);
2796 mnt_dev = strdup(tmp + strlen("dev="));
2797 if (mnt_dev)
2798 {
2799 tmp = strchr(mnt_dev, ',');
2800 if (tmp)
2801 *tmp = '\0';
2802 }
2803 }
2804 }
2805 // use strstr here to cover things fs types like "udf,iso9660"
2806 if (strstr(mnt_type, "iso9660") == 0)
2807 {
2808 /** @todo check whether we've already got the drive in our list! */
2809 if (i_validateDevice(mnt_dev, true))
2810 {
2811 ComObjPtr<Medium> hostDVDDriveObj;
2812 hostDVDDriveObj.createObject();
2813 hostDVDDriveObj->init(m->pParent, DeviceType_DVD, Bstr(mnt_dev));
2814 list.push_back (hostDVDDriveObj);
2815 }
2816 }
2817 free(mnt_dev);
2818 free(mnt_type);
2819 }
2820 endmntent(mtab);
2821 }
2822#else // RT_OS_SOLARIS
2823 FILE *mntFile = fopen(mountTable, "r");
2824 if (mntFile)
2825 {
2826 struct mnttab mntTab;
2827 while (getmntent(mntFile, &mntTab) == 0)
2828 {
2829 const char *mountName = mntTab.mnt_special;
2830 const char *mountPoint = mntTab.mnt_mountp;
2831 const char *mountFSType = mntTab.mnt_fstype;
2832 if (mountName && mountPoint && mountFSType)
2833 {
2834 // skip devices we are not interested in
2835 if ((*mountName && mountName[0] == '/') && // skip 'fake' devices (like -hosts,
2836 // proc, fd, swap)
2837 (*mountFSType && (strncmp(mountFSType, RT_STR_TUPLE("devfs")) != 0 && // skip devfs
2838 // (i.e. /devices)
2839 strncmp(mountFSType, RT_STR_TUPLE("dev")) != 0 && // skip dev (i.e. /dev)
2840 strncmp(mountFSType, RT_STR_TUPLE("lofs")) != 0))) // skip loop-back file-system (lofs)
2841 {
2842 char *rawDevName = getfullrawname((char *)mountName);
2843 if (i_validateDevice(rawDevName, true))
2844 {
2845 ComObjPtr<Medium> hostDVDDriveObj;
2846 hostDVDDriveObj.createObject();
2847 hostDVDDriveObj->init(m->pParent, DeviceType_DVD, Bstr(rawDevName));
2848 list.push_back(hostDVDDriveObj);
2849 }
2850 free(rawDevName);
2851 }
2852 }
2853 }
2854
2855 fclose(mntFile);
2856 }
2857#endif
2858}
2859
2860/**
2861 * Helper function to check whether the given device node is a valid drive
2862 */
2863bool Host::i_validateDevice(const char *deviceNode, bool isCDROM)
2864{
2865 struct stat statInfo;
2866 bool retValue = false;
2867
2868 // sanity check
2869 if (!deviceNode)
2870 {
2871 return false;
2872 }
2873
2874 // first a simple stat() call
2875 if (stat(deviceNode, &statInfo) < 0)
2876 {
2877 return false;
2878 }
2879 else
2880 {
2881 if (isCDROM)
2882 {
2883 if (S_ISCHR(statInfo.st_mode) || S_ISBLK(statInfo.st_mode))
2884 {
2885 int fileHandle;
2886 // now try to open the device
2887 fileHandle = open(deviceNode, O_RDONLY | O_NONBLOCK, 0);
2888 if (fileHandle >= 0)
2889 {
2890 cdrom_subchnl cdChannelInfo;
2891 cdChannelInfo.cdsc_format = CDROM_MSF;
2892 // this call will finally reveal the whole truth
2893#ifdef RT_OS_LINUX
2894 if ((ioctl(fileHandle, CDROMSUBCHNL, &cdChannelInfo) == 0) ||
2895 (errno == EIO) || (errno == ENOENT) ||
2896 (errno == EINVAL) || (errno == ENOMEDIUM))
2897#else
2898 if ((ioctl(fileHandle, CDROMSUBCHNL, &cdChannelInfo) == 0) ||
2899 (errno == EIO) || (errno == ENOENT) ||
2900 (errno == EINVAL))
2901#endif
2902 {
2903 retValue = true;
2904 }
2905 close(fileHandle);
2906 }
2907 }
2908 } else
2909 {
2910 // floppy case
2911 if (S_ISCHR(statInfo.st_mode) || S_ISBLK(statInfo.st_mode))
2912 {
2913 /// @todo do some more testing, maybe a nice IOCTL!
2914 retValue = true;
2915 }
2916 }
2917 }
2918 return retValue;
2919}
2920#endif // RT_OS_SOLARIS
2921
2922#ifdef VBOX_WITH_USB
2923/**
2924 * Checks for the presence and status of the USB Proxy Service.
2925 * Returns S_OK when the Proxy is present and OK, VBOX_E_HOST_ERROR (as a
2926 * warning) if the proxy service is not available due to the way the host is
2927 * configured (at present, that means that usbfs and hal/DBus are not
2928 * available on a Linux host) or E_FAIL and a corresponding error message
2929 * otherwise. Intended to be used by methods that rely on the Proxy Service
2930 * availability.
2931 *
2932 * @note This method may return a warning result code. It is recommended to use
2933 * MultiError to store the return value.
2934 *
2935 * @note Locks this object for reading.
2936 */
2937HRESULT Host::i_checkUSBProxyService()
2938{
2939 AutoCaller autoCaller(this);
2940 if (FAILED(autoCaller.rc()))
2941 return autoCaller.rc();
2942
2943 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2944
2945 AssertReturn(m->pUSBProxyService, E_FAIL);
2946 if (!m->pUSBProxyService->isActive())
2947 {
2948 /* disable the USB controller completely to avoid assertions if the
2949 * USB proxy service could not start. */
2950
2951 switch (m->pUSBProxyService->getLastError())
2952 {
2953 case VERR_FILE_NOT_FOUND: /** @todo what does this mean? */
2954 return setWarning(E_FAIL,
2955 tr("Could not load the Host USB Proxy Service (VERR_FILE_NOT_FOUND). The service might not be installed on the host computer"));
2956 case VERR_VUSB_USB_DEVICE_PERMISSION:
2957 return setWarning(E_FAIL,
2958 tr("VirtualBox is not currently allowed to access USB devices. You can change this by adding your user to the 'vboxusers' group. Please see the user manual for a more detailed explanation"));
2959 case VERR_VUSB_USBFS_PERMISSION:
2960 return setWarning(E_FAIL,
2961 tr("VirtualBox is not currently allowed to access USB devices. You can change this by allowing your user to access the 'usbfs' folder and files. Please see the user manual for a more detailed explanation"));
2962 case VINF_SUCCESS:
2963 return setWarning(E_FAIL,
2964 tr("The USB Proxy Service has not yet been ported to this host"));
2965 default:
2966 return setWarning(E_FAIL, "%s: %Rrc",
2967 tr("Could not load the Host USB Proxy service"),
2968 m->pUSBProxyService->getLastError());
2969 }
2970 }
2971
2972 return S_OK;
2973}
2974#endif /* VBOX_WITH_USB */
2975
2976HRESULT Host::i_updateNetIfList()
2977{
2978#ifdef VBOX_WITH_HOSTNETIF_API
2979 AssertReturn(!isWriteLockOnCurrentThread(), E_FAIL);
2980
2981 /** @todo r=klaus it would save lots of clock cycles if for concurrent
2982 * threads executing this code we'd only do one interface enumeration
2983 * and update, and let the other threads use the result as is. However
2984 * if there's a constant hammering of this method, we don't want this
2985 * to cause update starvation. */
2986 HostNetworkInterfaceList list;
2987 int rc = NetIfList(list);
2988 if (rc)
2989 {
2990 Log(("Failed to get host network interface list with rc=%Rrc\n", rc));
2991 return E_FAIL;
2992 }
2993
2994 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
2995
2996 AssertReturn(m->pParent, E_FAIL);
2997 /* Make a copy as the original may be partially destroyed later. */
2998 HostNetworkInterfaceList listCopy(list);
2999 HostNetworkInterfaceList::iterator itOld, itNew;
3000# ifdef VBOX_WITH_RESOURCE_USAGE_API
3001 PerformanceCollector *aCollector = m->pParent->i_performanceCollector();
3002# endif
3003 for (itOld = m->llNetIfs.begin(); itOld != m->llNetIfs.end(); ++itOld)
3004 {
3005 bool fGone = true;
3006 Bstr nameOld;
3007 (*itOld)->COMGETTER(Name)(nameOld.asOutParam());
3008 for (itNew = listCopy.begin(); itNew != listCopy.end(); ++itNew)
3009 {
3010 Bstr nameNew;
3011 (*itNew)->COMGETTER(Name)(nameNew.asOutParam());
3012 if (nameNew == nameOld)
3013 {
3014 fGone = false;
3015 (*itNew)->uninit();
3016 listCopy.erase(itNew);
3017 break;
3018 }
3019 }
3020 if (fGone)
3021 {
3022# ifdef VBOX_WITH_RESOURCE_USAGE_API
3023 (*itOld)->i_unregisterMetrics(aCollector, this);
3024 (*itOld)->uninit();
3025# endif
3026 }
3027 }
3028 /*
3029 * Need to set the references to VirtualBox object in all interface objects
3030 * (see @bugref{6439}).
3031 */
3032 for (itNew = list.begin(); itNew != list.end(); ++itNew)
3033 (*itNew)->i_setVirtualBox(m->pParent);
3034 /* At this point listCopy will contain newly discovered interfaces only. */
3035 for (itNew = listCopy.begin(); itNew != listCopy.end(); ++itNew)
3036 {
3037 HostNetworkInterfaceType_T t;
3038 HRESULT hrc = (*itNew)->COMGETTER(InterfaceType)(&t);
3039 if (FAILED(hrc))
3040 {
3041 Bstr n;
3042 (*itNew)->COMGETTER(Name)(n.asOutParam());
3043 LogRel(("Host::updateNetIfList: failed to get interface type for %ls\n", n.raw()));
3044 }
3045 else if (t == HostNetworkInterfaceType_Bridged)
3046 {
3047# ifdef VBOX_WITH_RESOURCE_USAGE_API
3048 (*itNew)->i_registerMetrics(aCollector, this);
3049# endif
3050 }
3051 }
3052 m->llNetIfs = list;
3053 return S_OK;
3054#else
3055 return E_NOTIMPL;
3056#endif
3057}
3058
3059#ifdef VBOX_WITH_RESOURCE_USAGE_API
3060
3061void Host::i_registerDiskMetrics(PerformanceCollector *aCollector)
3062{
3063 pm::CollectorHAL *hal = aCollector->getHAL();
3064 /* Create sub metrics */
3065 Utf8StrFmt fsNameBase("FS/{%s}/Usage", "/");
3066 //Utf8StrFmt fsNameBase("Filesystem/[root]/Usage");
3067 pm::SubMetric *fsRootUsageTotal = new pm::SubMetric(fsNameBase + "/Total",
3068 "Root file system size.");
3069 pm::SubMetric *fsRootUsageUsed = new pm::SubMetric(fsNameBase + "/Used",
3070 "Root file system space currently occupied.");
3071 pm::SubMetric *fsRootUsageFree = new pm::SubMetric(fsNameBase + "/Free",
3072 "Root file system space currently empty.");
3073
3074 pm::BaseMetric *fsRootUsage = new pm::HostFilesystemUsage(hal, this,
3075 fsNameBase, "/",
3076 fsRootUsageTotal,
3077 fsRootUsageUsed,
3078 fsRootUsageFree);
3079 aCollector->registerBaseMetric(fsRootUsage);
3080
3081 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageTotal, 0));
3082 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageTotal,
3083 new pm::AggregateAvg()));
3084 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageTotal,
3085 new pm::AggregateMin()));
3086 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageTotal,
3087 new pm::AggregateMax()));
3088
3089 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageUsed, 0));
3090 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageUsed,
3091 new pm::AggregateAvg()));
3092 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageUsed,
3093 new pm::AggregateMin()));
3094 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageUsed,
3095 new pm::AggregateMax()));
3096
3097 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageFree, 0));
3098 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageFree,
3099 new pm::AggregateAvg()));
3100 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageFree,
3101 new pm::AggregateMin()));
3102 aCollector->registerMetric(new pm::Metric(fsRootUsage, fsRootUsageFree,
3103 new pm::AggregateMax()));
3104
3105 /* For now we are concerned with the root file system only. */
3106 pm::DiskList disksUsage, disksLoad;
3107 int rc = hal->getDiskListByFs("/", disksUsage, disksLoad);
3108 if (RT_FAILURE(rc))
3109 return;
3110 pm::DiskList::iterator it;
3111 for (it = disksLoad.begin(); it != disksLoad.end(); ++it)
3112 {
3113 Utf8StrFmt strName("Disk/%s", it->c_str());
3114 pm::SubMetric *fsLoadUtil = new pm::SubMetric(strName + "/Load/Util",
3115 "Percentage of time disk was busy serving I/O requests.");
3116 pm::BaseMetric *fsLoad = new pm::HostDiskLoadRaw(hal, this, strName + "/Load",
3117 *it, fsLoadUtil);
3118 aCollector->registerBaseMetric(fsLoad);
3119
3120 aCollector->registerMetric(new pm::Metric(fsLoad, fsLoadUtil, 0));
3121 aCollector->registerMetric(new pm::Metric(fsLoad, fsLoadUtil,
3122 new pm::AggregateAvg()));
3123 aCollector->registerMetric(new pm::Metric(fsLoad, fsLoadUtil,
3124 new pm::AggregateMin()));
3125 aCollector->registerMetric(new pm::Metric(fsLoad, fsLoadUtil,
3126 new pm::AggregateMax()));
3127 }
3128 for (it = disksUsage.begin(); it != disksUsage.end(); ++it)
3129 {
3130 Utf8StrFmt strName("Disk/%s", it->c_str());
3131 pm::SubMetric *fsUsageTotal = new pm::SubMetric(strName + "/Usage/Total",
3132 "Disk size.");
3133 pm::BaseMetric *fsUsage = new pm::HostDiskUsage(hal, this, strName + "/Usage",
3134 *it, fsUsageTotal);
3135 aCollector->registerBaseMetric(fsUsage);
3136
3137 aCollector->registerMetric(new pm::Metric(fsUsage, fsUsageTotal, 0));
3138 aCollector->registerMetric(new pm::Metric(fsUsage, fsUsageTotal,
3139 new pm::AggregateAvg()));
3140 aCollector->registerMetric(new pm::Metric(fsUsage, fsUsageTotal,
3141 new pm::AggregateMin()));
3142 aCollector->registerMetric(new pm::Metric(fsUsage, fsUsageTotal,
3143 new pm::AggregateMax()));
3144 }
3145}
3146
3147void Host::i_registerMetrics(PerformanceCollector *aCollector)
3148{
3149 pm::CollectorHAL *hal = aCollector->getHAL();
3150 /* Create sub metrics */
3151 pm::SubMetric *cpuLoadUser = new pm::SubMetric("CPU/Load/User",
3152 "Percentage of processor time spent in user mode.");
3153 pm::SubMetric *cpuLoadKernel = new pm::SubMetric("CPU/Load/Kernel",
3154 "Percentage of processor time spent in kernel mode.");
3155 pm::SubMetric *cpuLoadIdle = new pm::SubMetric("CPU/Load/Idle",
3156 "Percentage of processor time spent idling.");
3157 pm::SubMetric *cpuMhzSM = new pm::SubMetric("CPU/MHz",
3158 "Average of current frequency of all processors.");
3159 pm::SubMetric *ramUsageTotal = new pm::SubMetric("RAM/Usage/Total",
3160 "Total physical memory installed.");
3161 pm::SubMetric *ramUsageUsed = new pm::SubMetric("RAM/Usage/Used",
3162 "Physical memory currently occupied.");
3163 pm::SubMetric *ramUsageFree = new pm::SubMetric("RAM/Usage/Free",
3164 "Physical memory currently available to applications.");
3165 pm::SubMetric *ramVMMUsed = new pm::SubMetric("RAM/VMM/Used",
3166 "Total physical memory used by the hypervisor.");
3167 pm::SubMetric *ramVMMFree = new pm::SubMetric("RAM/VMM/Free",
3168 "Total physical memory free inside the hypervisor.");
3169 pm::SubMetric *ramVMMBallooned = new pm::SubMetric("RAM/VMM/Ballooned",
3170 "Total physical memory ballooned by the hypervisor.");
3171 pm::SubMetric *ramVMMShared = new pm::SubMetric("RAM/VMM/Shared",
3172 "Total physical memory shared between VMs.");
3173
3174
3175 /* Create and register base metrics */
3176 pm::BaseMetric *cpuLoad = new pm::HostCpuLoadRaw(hal, this, cpuLoadUser, cpuLoadKernel,
3177 cpuLoadIdle);
3178 aCollector->registerBaseMetric(cpuLoad);
3179 pm::BaseMetric *cpuMhz = new pm::HostCpuMhz(hal, this, cpuMhzSM);
3180 aCollector->registerBaseMetric(cpuMhz);
3181 pm::BaseMetric *ramUsage = new pm::HostRamUsage(hal, this,
3182 ramUsageTotal,
3183 ramUsageUsed,
3184 ramUsageFree);
3185 aCollector->registerBaseMetric(ramUsage);
3186 pm::BaseMetric *ramVmm = new pm::HostRamVmm(aCollector->getGuestManager(), this,
3187 ramVMMUsed,
3188 ramVMMFree,
3189 ramVMMBallooned,
3190 ramVMMShared);
3191 aCollector->registerBaseMetric(ramVmm);
3192
3193 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadUser, 0));
3194 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadUser,
3195 new pm::AggregateAvg()));
3196 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadUser,
3197 new pm::AggregateMin()));
3198 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadUser,
3199 new pm::AggregateMax()));
3200
3201 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadKernel, 0));
3202 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadKernel,
3203 new pm::AggregateAvg()));
3204 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadKernel,
3205 new pm::AggregateMin()));
3206 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadKernel,
3207 new pm::AggregateMax()));
3208
3209 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadIdle, 0));
3210 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadIdle,
3211 new pm::AggregateAvg()));
3212 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadIdle,
3213 new pm::AggregateMin()));
3214 aCollector->registerMetric(new pm::Metric(cpuLoad, cpuLoadIdle,
3215 new pm::AggregateMax()));
3216
3217 aCollector->registerMetric(new pm::Metric(cpuMhz, cpuMhzSM, 0));
3218 aCollector->registerMetric(new pm::Metric(cpuMhz, cpuMhzSM,
3219 new pm::AggregateAvg()));
3220 aCollector->registerMetric(new pm::Metric(cpuMhz, cpuMhzSM,
3221 new pm::AggregateMin()));
3222 aCollector->registerMetric(new pm::Metric(cpuMhz, cpuMhzSM,
3223 new pm::AggregateMax()));
3224
3225 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageTotal, 0));
3226 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageTotal,
3227 new pm::AggregateAvg()));
3228 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageTotal,
3229 new pm::AggregateMin()));
3230 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageTotal,
3231 new pm::AggregateMax()));
3232
3233 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageUsed, 0));
3234 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageUsed,
3235 new pm::AggregateAvg()));
3236 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageUsed,
3237 new pm::AggregateMin()));
3238 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageUsed,
3239 new pm::AggregateMax()));
3240
3241 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageFree, 0));
3242 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageFree,
3243 new pm::AggregateAvg()));
3244 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageFree,
3245 new pm::AggregateMin()));
3246 aCollector->registerMetric(new pm::Metric(ramUsage, ramUsageFree,
3247 new pm::AggregateMax()));
3248
3249 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMUsed, 0));
3250 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMUsed,
3251 new pm::AggregateAvg()));
3252 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMUsed,
3253 new pm::AggregateMin()));
3254 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMUsed,
3255 new pm::AggregateMax()));
3256
3257 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMFree, 0));
3258 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMFree,
3259 new pm::AggregateAvg()));
3260 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMFree,
3261 new pm::AggregateMin()));
3262 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMFree,
3263 new pm::AggregateMax()));
3264
3265 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMBallooned, 0));
3266 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMBallooned,
3267 new pm::AggregateAvg()));
3268 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMBallooned,
3269 new pm::AggregateMin()));
3270 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMBallooned,
3271 new pm::AggregateMax()));
3272
3273 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMShared, 0));
3274 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMShared,
3275 new pm::AggregateAvg()));
3276 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMShared,
3277 new pm::AggregateMin()));
3278 aCollector->registerMetric(new pm::Metric(ramVmm, ramVMMShared,
3279 new pm::AggregateMax()));
3280 i_registerDiskMetrics(aCollector);
3281}
3282
3283void Host::i_unregisterMetrics(PerformanceCollector *aCollector)
3284{
3285 aCollector->unregisterMetricsFor(this);
3286 aCollector->unregisterBaseMetricsFor(this);
3287}
3288
3289#endif /* VBOX_WITH_RESOURCE_USAGE_API */
3290
3291
3292/* static */
3293void Host::i_generateMACAddress(Utf8Str &mac)
3294{
3295 /*
3296 * Our strategy is as follows: the first three bytes are our fixed
3297 * vendor ID (080027). The remaining 3 bytes will be taken from the
3298 * start of a GUID. This is a fairly safe algorithm.
3299 */
3300 Guid guid;
3301 guid.create();
3302 mac = Utf8StrFmt("080027%02X%02X%02X",
3303 guid.raw()->au8[0], guid.raw()->au8[1], guid.raw()->au8[2]);
3304}
3305
3306/* vi: set tabstop=4 shiftwidth=4 expandtab: */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette