VirtualBox

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

Last change on this file since 39789 was 39429, checked in by vboxsync, 13 years ago

Main/NetIf: Fixed segmentation fault in VBoxSVC on host-creation failure

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