VirtualBox

source: vbox/trunk/src/VBox/Main/HostImpl.cpp@ 12986

Last change on this file since 12986 was 12954, checked in by vboxsync, 16 years ago

Simpler.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 106.6 KB
Line 
1/* $Id: HostImpl.cpp 12954 2008-10-02 18:02:31Z vboxsync $ */
2/** @file
3 * VirtualBox COM class implementation: Host
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22#define __STDC_LIMIT_MACROS
23#define __STDC_CONSTANT_MACROS
24
25#ifdef RT_OS_LINUX
26# include <sys/types.h>
27# include <sys/stat.h>
28# include <unistd.h>
29# include <sys/ioctl.h>
30# include <fcntl.h>
31# include <mntent.h>
32/* bird: This is a hack to work around conflicts between these linux kernel headers
33 * and the GLIBC tcpip headers. They have different declarations of the 4
34 * standard byte order functions. */
35# define _LINUX_BYTEORDER_GENERIC_H
36# include <linux/cdrom.h>
37# ifdef VBOX_USE_LIBHAL
38// # include <libhal.h>
39// /* These are defined by libhal.h and by VBox header files. */
40// # undef TRUE
41// # undef FALSE
42# include "vbox-libhal.h"
43# endif
44# include <errno.h>
45#endif /* RT_OS_LINUX */
46
47#ifdef RT_OS_SOLARIS
48# include <fcntl.h>
49# include <unistd.h>
50# include <stropts.h>
51# include <errno.h>
52# include <limits.h>
53# include <stdio.h>
54# ifdef VBOX_SOLARIS_NSL_RESOLVED
55# include <libdevinfo.h>
56# else
57# include <net/if.h>
58# include <sys/socket.h>
59# include <sys/sockio.h>
60# include <net/if_arp.h>
61# include <net/if.h>
62# endif /* VBOX_SOLARIS_USE_DEVINFO */
63# include <sys/types.h>
64# include <sys/stat.h>
65# include <sys/cdio.h>
66# include <sys/dkio.h>
67# include <sys/mnttab.h>
68# include <sys/mntent.h>
69# ifdef VBOX_USE_LIBHAL
70# include "vbox-libhal.h"
71extern "C" char *getfullrawname(char *);
72# endif
73# include "solaris/DynLoadLibSolaris.h"
74#endif /* RT_OS_SOLARIS */
75
76#ifdef RT_OS_WINDOWS
77# define _WIN32_DCOM
78# include <windows.h>
79# include <shellapi.h>
80# define INITGUID
81# include <guiddef.h>
82# include <devguid.h>
83# include <objbase.h>
84# include <setupapi.h>
85# include <shlobj.h>
86# include <cfgmgr32.h>
87
88#endif /* RT_OS_WINDOWS */
89
90
91#include "HostImpl.h"
92#include "HostDVDDriveImpl.h"
93#include "HostFloppyDriveImpl.h"
94#include "HostNetworkInterfaceImpl.h"
95#ifdef VBOX_WITH_USB
96# include "HostUSBDeviceImpl.h"
97# include "USBDeviceFilterImpl.h"
98# include "USBProxyService.h"
99#endif
100#include "VirtualBoxImpl.h"
101#include "MachineImpl.h"
102#include "Logging.h"
103
104#ifdef RT_OS_DARWIN
105# include "darwin/iokit.h"
106#endif
107
108
109#include <VBox/usb.h>
110#include <VBox/err.h>
111#include <iprt/string.h>
112#include <iprt/mp.h>
113#include <iprt/time.h>
114#include <iprt/param.h>
115#include <iprt/env.h>
116#ifdef RT_OS_SOLARIS
117# include <iprt/path.h>
118# include <iprt/ctype.h>
119#endif
120
121#include <stdio.h>
122
123#include <algorithm>
124
125// constructor / destructor
126/////////////////////////////////////////////////////////////////////////////
127
128HRESULT Host::FinalConstruct()
129{
130 return S_OK;
131}
132
133void Host::FinalRelease()
134{
135 if (isReady())
136 uninit();
137}
138
139// public initializer/uninitializer for internal purposes only
140/////////////////////////////////////////////////////////////////////////////
141
142/**
143 * Initializes the host object.
144 *
145 * @param aParent VirtualBox parent object.
146 */
147HRESULT Host::init (VirtualBox *aParent)
148{
149 LogFlowThisFunc (("isReady=%d\n", isReady()));
150
151 ComAssertRet (aParent, E_INVALIDARG);
152
153 AutoWriteLock alock (this);
154 ComAssertRet (!isReady(), E_UNEXPECTED);
155
156 mParent = aParent;
157
158#ifdef VBOX_WITH_USB
159 /*
160 * Create and initialize the USB Proxy Service.
161 */
162# if defined (RT_OS_DARWIN)
163 mUSBProxyService = new USBProxyServiceDarwin (this);
164# elif defined (RT_OS_LINUX)
165 mUSBProxyService = new USBProxyServiceLinux (this);
166# elif defined (RT_OS_OS2)
167 mUSBProxyService = new USBProxyServiceOs2 (this);
168# elif defined (RT_OS_SOLARIS) && 0
169 mUSBProxyService = new USBProxyServiceSolaris (this);
170# elif defined (RT_OS_WINDOWS)
171 mUSBProxyService = new USBProxyServiceWindows (this);
172# else
173 mUSBProxyService = new USBProxyService (this);
174# endif
175 HRESULT hrc = mUSBProxyService->init();
176 AssertComRCReturn(hrc, hrc);
177#endif /* VBOX_WITH_USB */
178
179#ifdef VBOX_WITH_RESOURCE_USAGE_API
180 registerMetrics (aParent->performanceCollector());
181#endif /* VBOX_WITH_RESOURCE_USAGE_API */
182
183 setReady(true);
184 return S_OK;
185}
186
187/**
188 * Uninitializes the host object and sets the ready flag to FALSE.
189 * Called either from FinalRelease() or by the parent when it gets destroyed.
190 */
191void Host::uninit()
192{
193 LogFlowThisFunc (("isReady=%d\n", isReady()));
194
195 AssertReturn (isReady(), (void) 0);
196
197#ifdef VBOX_WITH_RESOURCE_USAGE_API
198 unregisterMetrics (mParent->performanceCollector());
199#endif /* VBOX_WITH_RESOURCE_USAGE_API */
200
201#ifdef VBOX_WITH_USB
202 /* wait for USB proxy service to terminate before we uninit all USB
203 * devices */
204 LogFlowThisFunc (("Stopping USB proxy service...\n"));
205 delete mUSBProxyService;
206 mUSBProxyService = NULL;
207 LogFlowThisFunc (("Done stopping USB proxy service.\n"));
208#endif
209
210 /* uninit all USB device filters still referenced by clients */
211 uninitDependentChildren();
212
213#ifdef VBOX_WITH_USB
214 mUSBDeviceFilters.clear();
215#endif
216
217 setReady (FALSE);
218}
219
220// IHost properties
221/////////////////////////////////////////////////////////////////////////////
222
223/**
224 * Returns a list of host DVD drives.
225 *
226 * @returns COM status code
227 * @param drives address of result pointer
228 */
229STDMETHODIMP Host::COMGETTER(DVDDrives) (IHostDVDDriveCollection **drives)
230{
231 if (!drives)
232 return E_POINTER;
233 AutoWriteLock alock (this);
234 CHECK_READY();
235 std::list <ComObjPtr <HostDVDDrive> > list;
236
237#if defined(RT_OS_WINDOWS)
238 int sz = GetLogicalDriveStrings(0, NULL);
239 TCHAR *hostDrives = new TCHAR[sz+1];
240 GetLogicalDriveStrings(sz, hostDrives);
241 wchar_t driveName[3] = { '?', ':', '\0' };
242 TCHAR *p = hostDrives;
243 do
244 {
245 if (GetDriveType(p) == DRIVE_CDROM)
246 {
247 driveName[0] = *p;
248 ComObjPtr <HostDVDDrive> hostDVDDriveObj;
249 hostDVDDriveObj.createObject();
250 hostDVDDriveObj->init (Bstr (driveName));
251 list.push_back (hostDVDDriveObj);
252 }
253 p += _tcslen(p) + 1;
254 }
255 while (*p);
256 delete[] hostDrives;
257
258#elif defined(RT_OS_SOLARIS)
259# ifdef VBOX_USE_LIBHAL
260 if (!getDVDInfoFromHal(list))
261# endif
262 // Not all Solaris versions ship with libhal.
263 // So use a fallback approach similar to Linux.
264 {
265 if (RTEnvGet("VBOX_CDROM"))
266 {
267 char *cdromEnv = strdup(RTEnvGet("VBOX_CDROM"));
268 char *cdromDrive;
269 cdromDrive = strtok(cdromEnv, ":"); /** @todo use strtok_r. */
270 while (cdromDrive)
271 {
272 if (validateDevice(cdromDrive, true))
273 {
274 ComObjPtr <HostDVDDrive> hostDVDDriveObj;
275 hostDVDDriveObj.createObject();
276 hostDVDDriveObj->init (Bstr (cdromDrive));
277 list.push_back (hostDVDDriveObj);
278 }
279 cdromDrive = strtok(NULL, ":");
280 }
281 free(cdromEnv);
282 }
283 else
284 {
285 // this might work on Solaris version older than Nevada.
286 if (validateDevice("/cdrom/cdrom0", true))
287 {
288 ComObjPtr <HostDVDDrive> hostDVDDriveObj;
289 hostDVDDriveObj.createObject();
290 hostDVDDriveObj->init (Bstr ("cdrom/cdrom0"));
291 list.push_back (hostDVDDriveObj);
292 }
293
294 // check the mounted drives
295 parseMountTable(MNTTAB, list);
296 }
297 }
298
299#elif defined(RT_OS_LINUX)
300#ifdef VBOX_USE_LIBHAL
301 if (!getDVDInfoFromHal(list)) /* Playing with #defines in this way is nasty, I know. */
302#endif /* USE_LIBHAL defined */
303 // On Linux without hal, the situation is much more complex. We will take a
304 // heuristical approach and also allow the user to specify a list of host
305 // CDROMs using an environment variable.
306 // The general strategy is to try some known device names and see of they
307 // exist. At last, we'll enumerate the /etc/fstab file (luckily there's an
308 // API to parse it) for CDROM devices. Ok, let's start!
309
310 {
311 if (RTEnvGet("VBOX_CDROM"))
312 {
313 char *cdromEnv = strdupa(RTEnvGet("VBOX_CDROM"));
314 char *cdromDrive;
315 cdromDrive = strtok(cdromEnv, ":"); /** @todo use strtok_r */
316 while (cdromDrive)
317 {
318 if (validateDevice(cdromDrive, true))
319 {
320 ComObjPtr <HostDVDDrive> hostDVDDriveObj;
321 hostDVDDriveObj.createObject();
322 hostDVDDriveObj->init (Bstr (cdromDrive));
323 list.push_back (hostDVDDriveObj);
324 }
325 cdromDrive = strtok(NULL, ":");
326 }
327 }
328 else
329 {
330 // this is a good guess usually
331 if (validateDevice("/dev/cdrom", true))
332 {
333 ComObjPtr <HostDVDDrive> hostDVDDriveObj;
334 hostDVDDriveObj.createObject();
335 hostDVDDriveObj->init (Bstr ("/dev/cdrom"));
336 list.push_back (hostDVDDriveObj);
337 }
338
339 // check the mounted drives
340 parseMountTable((char*)"/etc/mtab", list);
341
342 // check the drives that can be mounted
343 parseMountTable((char*)"/etc/fstab", list);
344 }
345 }
346#elif defined(RT_OS_DARWIN)
347 PDARWINDVD cur = DarwinGetDVDDrives();
348 while (cur)
349 {
350 ComObjPtr<HostDVDDrive> hostDVDDriveObj;
351 hostDVDDriveObj.createObject();
352 hostDVDDriveObj->init(Bstr(cur->szName));
353 list.push_back(hostDVDDriveObj);
354
355 /* next */
356 void *freeMe = cur;
357 cur = cur->pNext;
358 RTMemFree(freeMe);
359 }
360
361#else
362 /* PORTME */
363#endif
364
365 ComObjPtr<HostDVDDriveCollection> collection;
366 collection.createObject();
367 collection->init (list);
368 collection.queryInterfaceTo(drives);
369 return S_OK;
370}
371
372/**
373 * Returns a list of host floppy drives.
374 *
375 * @returns COM status code
376 * @param drives address of result pointer
377 */
378STDMETHODIMP Host::COMGETTER(FloppyDrives) (IHostFloppyDriveCollection **drives)
379{
380 if (!drives)
381 return E_POINTER;
382 AutoWriteLock alock (this);
383 CHECK_READY();
384
385 std::list <ComObjPtr <HostFloppyDrive> > list;
386
387#ifdef RT_OS_WINDOWS
388 int sz = GetLogicalDriveStrings(0, NULL);
389 TCHAR *hostDrives = new TCHAR[sz+1];
390 GetLogicalDriveStrings(sz, hostDrives);
391 wchar_t driveName[3] = { '?', ':', '\0' };
392 TCHAR *p = hostDrives;
393 do
394 {
395 if (GetDriveType(p) == DRIVE_REMOVABLE)
396 {
397 driveName[0] = *p;
398 ComObjPtr <HostFloppyDrive> hostFloppyDriveObj;
399 hostFloppyDriveObj.createObject();
400 hostFloppyDriveObj->init (Bstr (driveName));
401 list.push_back (hostFloppyDriveObj);
402 }
403 p += _tcslen(p) + 1;
404 }
405 while (*p);
406 delete[] hostDrives;
407#elif defined(RT_OS_LINUX)
408#ifdef VBOX_USE_LIBHAL
409 if (!getFloppyInfoFromHal(list)) /* Playing with #defines in this way is nasty, I know. */
410#endif /* USE_LIBHAL defined */
411 // As with the CDROMs, on Linux we have to take a multi-level approach
412 // involving parsing the mount tables. As this is not bulletproof, we'll
413 // give the user the chance to override the detection by an environment
414 // variable and skip the detection.
415
416 {
417 if (RTEnvGet("VBOX_FLOPPY"))
418 {
419 char *floppyEnv = strdupa(RTEnvGet("VBOX_FLOPPY"));
420 char *floppyDrive;
421 floppyDrive = strtok(floppyEnv, ":");
422 while (floppyDrive)
423 {
424 // check if this is an acceptable device
425 if (validateDevice(floppyDrive, false))
426 {
427 ComObjPtr <HostFloppyDrive> hostFloppyDriveObj;
428 hostFloppyDriveObj.createObject();
429 hostFloppyDriveObj->init (Bstr (floppyDrive));
430 list.push_back (hostFloppyDriveObj);
431 }
432 floppyDrive = strtok(NULL, ":");
433 }
434 }
435 else
436 {
437 // we assume that a floppy is always /dev/fd[x] with x from 0 to 7
438 char devName[10];
439 for (int i = 0; i <= 7; i++)
440 {
441 sprintf(devName, "/dev/fd%d", i);
442 if (validateDevice(devName, false))
443 {
444 ComObjPtr <HostFloppyDrive> hostFloppyDriveObj;
445 hostFloppyDriveObj.createObject();
446 hostFloppyDriveObj->init (Bstr (devName));
447 list.push_back (hostFloppyDriveObj);
448 }
449 }
450 }
451 }
452#else
453 /* PORTME */
454#endif
455
456 ComObjPtr<HostFloppyDriveCollection> collection;
457 collection.createObject();
458 collection->init (list);
459 collection.queryInterfaceTo(drives);
460 return S_OK;
461}
462
463#ifdef RT_OS_WINDOWS
464/**
465 * Windows helper function for Host::COMGETTER(NetworkInterfaces).
466 *
467 * @returns true / false.
468 *
469 * @param guid The GUID.
470 */
471static bool IsTAPDevice(const char *guid)
472{
473 HKEY hNetcard;
474 LONG status;
475 DWORD len;
476 int i = 0;
477 bool ret = false;
478
479 status = RegOpenKeyExA(HKEY_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Control\\Class\\{4D36E972-E325-11CE-BFC1-08002BE10318}", 0, KEY_READ, &hNetcard);
480 if (status != ERROR_SUCCESS)
481 return false;
482
483 for (;;)
484 {
485 char szEnumName[256];
486 char szNetCfgInstanceId[256];
487 DWORD dwKeyType;
488 HKEY hNetCardGUID;
489
490 len = sizeof(szEnumName);
491 status = RegEnumKeyExA(hNetcard, i, szEnumName, &len, NULL, NULL, NULL, NULL);
492 if (status != ERROR_SUCCESS)
493 break;
494
495 status = RegOpenKeyExA(hNetcard, szEnumName, 0, KEY_READ, &hNetCardGUID);
496 if (status == ERROR_SUCCESS)
497 {
498 len = sizeof(szNetCfgInstanceId);
499 status = RegQueryValueExA(hNetCardGUID, "NetCfgInstanceId", NULL, &dwKeyType, (LPBYTE)szNetCfgInstanceId, &len);
500 if (status == ERROR_SUCCESS && dwKeyType == REG_SZ)
501 {
502 char szNetProductName[256];
503 char szNetProviderName[256];
504
505 szNetProductName[0] = 0;
506 len = sizeof(szNetProductName);
507 status = RegQueryValueExA(hNetCardGUID, "ProductName", NULL, &dwKeyType, (LPBYTE)szNetProductName, &len);
508
509 szNetProviderName[0] = 0;
510 len = sizeof(szNetProviderName);
511 status = RegQueryValueExA(hNetCardGUID, "ProviderName", NULL, &dwKeyType, (LPBYTE)szNetProviderName, &len);
512
513 if ( !strcmp(szNetCfgInstanceId, guid)
514 && !strcmp(szNetProductName, "VirtualBox TAP Adapter")
515 && ( !strcmp(szNetProviderName, "innotek GmbH")
516 || !strcmp(szNetProviderName, "Sun Microsystems, Inc.")))
517 {
518 ret = true;
519 RegCloseKey(hNetCardGUID);
520 break;
521 }
522 }
523 RegCloseKey(hNetCardGUID);
524 }
525 ++i;
526 }
527
528 RegCloseKey(hNetcard);
529 return ret;
530}
531#endif /* RT_OS_WINDOWS */
532
533#ifdef RT_OS_SOLARIS
534static void vboxSolarisAddHostIface(char *pszIface, int Instance, PCRTMAC pMac, void *pvHostNetworkInterfaceList)
535{
536 std::list<ComObjPtr <HostNetworkInterface> > *pList = (std::list<ComObjPtr <HostNetworkInterface> > *)pvHostNetworkInterfaceList;
537 Assert(pList);
538
539 typedef std::map <std::string, std::string> NICMap;
540 typedef std::pair <std::string, std::string> NICPair;
541 static NICMap SolarisNICMap;
542 if (SolarisNICMap.empty())
543 {
544 SolarisNICMap.insert(NICPair("afe", "ADMtek Centaur/Comet Fast Ethernet"));
545 SolarisNICMap.insert(NICPair("aggr", "Link Aggregation Interface"));
546 SolarisNICMap.insert(NICPair("bge", "Broadcom BCM57xx Gigabit Ethernet"));
547 SolarisNICMap.insert(NICPair("ce", "Cassini Gigabit Ethernet"));
548 SolarisNICMap.insert(NICPair("chxge", "Chelsio Ethernet"));
549 SolarisNICMap.insert(NICPair("dmfe", "Davicom Fast Ethernet"));
550 SolarisNICMap.insert(NICPair("dnet", "DEC 21040/41 21140 Ethernet"));
551 SolarisNICMap.insert(NICPair("e1000", "Intel PRO/1000 Gigabit Ethernet"));
552 SolarisNICMap.insert(NICPair("e1000g", "Intel PRO/1000 Gigabit Ethernet"));
553 SolarisNICMap.insert(NICPair("elx", "3COM EtherLink III Ethernet"));
554 SolarisNICMap.insert(NICPair("elxl", "3COM Ethernet"));
555 SolarisNICMap.insert(NICPair("eri", "eri Fast Ethernet"));
556 SolarisNICMap.insert(NICPair("ge", "GEM Gigabit Ethernet"));
557 SolarisNICMap.insert(NICPair("hme", "SUNW,hme Fast-Ethernet"));
558 SolarisNICMap.insert(NICPair("ipge", "PCI-E Gigabit Ethernet"));
559 SolarisNICMap.insert(NICPair("iprb", "Intel 82557/58/59 Ethernet"));
560 SolarisNICMap.insert(NICPair("mxfe", "Macronix 98715 Fast Ethernet"));
561 SolarisNICMap.insert(NICPair("nge", "Nvidia Gigabit Ethernet"));
562 SolarisNICMap.insert(NICPair("pcelx", "3COM EtherLink III PCMCIA Ethernet"));
563 SolarisNICMap.insert(NICPair("pcn", "AMD PCnet Ethernet"));
564 SolarisNICMap.insert(NICPair("qfe", "SUNW,qfe Quad Fast-Ethernet"));
565 SolarisNICMap.insert(NICPair("rge", "Realtek Gigabit Ethernet"));
566 SolarisNICMap.insert(NICPair("rtls", "Realtek 8139 Fast Ethernet"));
567 SolarisNICMap.insert(NICPair("skge", "SksKonnect Gigabit Ethernet"));
568 SolarisNICMap.insert(NICPair("spwr", "SMC EtherPower II 10/100 (9432) Ethernet"));
569 SolarisNICMap.insert(NICPair("vnic", "Virtual Network Interface Ethernet"));
570 SolarisNICMap.insert(NICPair("xge", "Neterior Xframe Gigabit Ethernet"));
571 SolarisNICMap.insert(NICPair("xge", "Neterior Xframe 10Gigabit Ethernet"));
572 }
573
574 /*
575 * Try picking up description from our NIC map.
576 */
577 char szNICInstance[128];
578 RTStrPrintf(szNICInstance, sizeof(szNICInstance), "%s%d", pszIface, Instance);
579 char szNICDesc[256];
580 std::string Description = SolarisNICMap[pszIface];
581 if (Description != "")
582 RTStrPrintf(szNICDesc, sizeof(szNICDesc), "%s - %s", szNICInstance, Description.c_str());
583 else
584 RTStrPrintf(szNICDesc, sizeof(szNICDesc), "%s - Ethernet", szNICInstance);
585
586 /*
587 * Construct UUID with interface name and the MAC address if available.
588 */
589 RTUUID Uuid;
590 RTUuidClear(&Uuid);
591 memcpy(&Uuid, szNICInstance, RT_MIN(strlen(szNICInstance), sizeof(Uuid)));
592 Uuid.Gen.u8ClockSeqHiAndReserved = (Uuid.Gen.u8ClockSeqHiAndReserved & 0x3f) | 0x80;
593 Uuid.Gen.u16TimeHiAndVersion = (Uuid.Gen.u16TimeHiAndVersion & 0x0fff) | 0x4000;
594 if (pMac)
595 {
596 Uuid.Gen.au8Node[0] = pMac->au8[0];
597 Uuid.Gen.au8Node[1] = pMac->au8[1];
598 Uuid.Gen.au8Node[2] = pMac->au8[2];
599 Uuid.Gen.au8Node[3] = pMac->au8[3];
600 Uuid.Gen.au8Node[4] = pMac->au8[4];
601 Uuid.Gen.au8Node[5] = pMac->au8[5];
602 }
603
604 ComObjPtr<HostNetworkInterface> IfObj;
605 IfObj.createObject();
606 if (SUCCEEDED(IfObj->init(Bstr(szNICDesc), Guid(Uuid))))
607 pList->push_back(IfObj);
608}
609
610static boolean_t vboxSolarisAddLinkHostIface(const char *pszIface, void *pvHostNetworkInterfaceList)
611{
612 /*
613 * Clip off the instance number from the interface name.
614 */
615 int cbInstance = 0;
616 int cbIface = strlen(pszIface);
617 const char *pszEnd = pszIface + cbIface - 1;
618 for (int i = 0; i < cbIface - 1; i++)
619 {
620 if (!RT_C_IS_DIGIT(*pszEnd))
621 break;
622 cbInstance++;
623 pszEnd--;
624 }
625
626 /*
627 * Add the interface.
628 */
629 char szIfaceName[128];
630 strncpy(szIfaceName, pszIface, cbIface - cbInstance);
631 szIfaceName[cbIface - cbInstance] = '\0';
632 int Instance = atoi(pszEnd + 1);
633 vboxSolarisAddHostIface(szIfaceName, Instance, NULL, pvHostNetworkInterfaceList);
634
635 /*
636 * Continue walking...
637 */
638 return _B_FALSE;
639}
640
641# ifdef VBOX_SOLARIS_NSL_RESOLVED
642static int vboxSolarisAddPhysHostIface(di_node_t Node, di_minor_t Minor, void *pvHostNetworkInterfaceList)
643{
644 /*
645 * Skip aggregations.
646 */
647 if (!strcmp(di_driver_name(Node), "aggr"))
648 return DI_WALK_CONTINUE;
649
650 /*
651 * Skip softmacs.
652 */
653 if (!strcmp(di_driver_name(Node), "softmac"))
654 return DI_WALK_CONTINUE;
655
656 vboxSolarisAddHostIface(di_driver_name(Node), di_instance(Node), NULL, pvHostNetworkInterfaceList);
657 return DI_WALK_CONTINUE;
658}
659
660static bool vboxSolarisSameNIC(ComObjPtr <HostNetworkInterface> Iface1, ComObjPtr <HostNetworkInterface> Iface2)
661{
662 Bstr Iface1Str;
663 (*Iface1).COMGETTER(Name) (Iface1Str.asOutParam());
664
665 Bstr Iface2Str;
666 (*Iface2).COMGETTER(Name) (Iface2Str.asOutParam());
667
668 return (Iface1Str == Iface2Str);
669}
670
671# endif /* VBOX_SOLARIS_USE_DEVINFO */
672
673#endif
674
675/**
676 * Returns a list of host network interfaces.
677 *
678 * @returns COM status code
679 * @param drives address of result pointer
680 */
681STDMETHODIMP Host::COMGETTER(NetworkInterfaces) (IHostNetworkInterfaceCollection **networkInterfaces)
682{
683#if defined(RT_OS_WINDOWS) || defined(VBOX_WITH_NETFLT) /*|| defined(RT_OS_OS2)*/
684 if (!networkInterfaces)
685 return E_POINTER;
686 AutoWriteLock alock (this);
687 CHECK_READY();
688
689 std::list <ComObjPtr <HostNetworkInterface> > list;
690
691# if defined(RT_OS_DARWIN)
692 PDARWINETHERNIC pEtherNICs = DarwinGetEthernetControllers();
693 while (pEtherNICs)
694 {
695 ComObjPtr<HostNetworkInterface> IfObj;
696 IfObj.createObject();
697 if (SUCCEEDED(IfObj->init(Bstr(pEtherNICs->szName), Guid(pEtherNICs->Uuid))))
698 list.push_back(IfObj);
699
700 /* next, free current */
701 void *pvFree = pEtherNICs;
702 pEtherNICs = pEtherNICs->pNext;
703 RTMemFree(pvFree);
704 }
705
706# elif defined(RT_OS_SOLARIS)
707
708#ifdef VBOX_SOLARIS_NSL_RESOLVED
709
710 /*
711 * Use libdevinfo for determining all physical interfaces.
712 */
713 di_node_t Root;
714 Root = di_init("/", DINFOCACHE);
715 if (Root != DI_NODE_NIL)
716 {
717 di_walk_minor(Root, DDI_NT_NET, 0, &list, vboxSolarisAddPhysHostIface);
718 di_fini(Root);
719 }
720
721 /*
722 * Use libdlpi for determining all DLPI interfaces.
723 */
724 if (VBoxSolarisLibDlpiFound())
725 g_pfnLibDlpiWalk(vboxSolarisAddLinkHostIface, &list, 0);
726
727 /*
728 * Weed out duplicates caused by dlpi_walk inconsistencies across Nevadas.
729 */
730 list.unique(vboxSolarisSameNIC);
731
732#else
733 /*
734 * This gets only the list of all plumbed logical interfaces.
735 */
736 int Sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_IP);
737 if (Sock > 0)
738 {
739 struct lifnum IfNum;
740 memset(&IfNum, 0, sizeof(IfNum));
741 IfNum.lifn_family = AF_INET;
742 int rc = ioctl(Sock, SIOCGLIFNUM, &IfNum);
743 if (!rc)
744 {
745 struct lifreq Ifaces[24];
746 struct lifconf IfConfig;
747 memset(&IfConfig, 0, sizeof(IfConfig));
748 IfConfig.lifc_family = AF_INET;
749 IfConfig.lifc_len = sizeof(Ifaces);
750 IfConfig.lifc_buf = (caddr_t)&(Ifaces[0]);
751 rc = ioctl(Sock, SIOCGLIFCONF, &IfConfig);
752 if (!rc)
753 {
754 /*
755 * Ok now we go the interfaces, get the info we need (i.e MAC address).
756 */
757 for (int i = 0; i < IfNum.lifn_count; i++)
758 {
759 /*
760 * Skip loopback interfaces.
761 */
762 if (!strncmp(Ifaces[i].lifr_name, "lo", 2))
763 continue;
764
765 rc = ioctl(Sock, SIOCGLIFADDR, &(Ifaces[i]));
766 if (!rc)
767 {
768 RTMAC Mac;
769 struct arpreq ArpReq;
770 memcpy(&ArpReq.arp_pa, &Ifaces[i].lifr_addr, sizeof(struct sockaddr_in));
771
772 /*
773 * We might fail if the interface has not been assigned an IP address.
774 * That doesn't matter; as long as it's plumbed we can pick it up.
775 * But, if it has not acquired an IP address we cannot obtain it's MAC
776 * address this way, so we just use all zeros there.
777 */
778 rc = ioctl(Sock, SIOCGARP, &ArpReq);
779 if (!rc)
780 memcpy(&Mac, ArpReq.arp_ha.sa_data, sizeof(RTMAC));
781 else
782 memset(&Mac, 0, sizeof(Mac));
783
784 char szNICDesc[LIFNAMSIZ + 256];
785 char *pszIface = Ifaces[i].lifr_name;
786 strcpy(szNICDesc, pszIface);
787
788 vboxSolarisAddLinkHostIface(pszIface, &list);
789 }
790 }
791 }
792 }
793 close(Sock);
794 }
795#endif
796
797# elif defined RT_OS_WINDOWS
798 static const char *NetworkKey = "SYSTEM\\CurrentControlSet\\Control\\Network\\"
799 "{4D36E972-E325-11CE-BFC1-08002BE10318}";
800 HKEY hCtrlNet;
801 LONG status;
802 DWORD len;
803 status = RegOpenKeyExA (HKEY_LOCAL_MACHINE, NetworkKey, 0, KEY_READ, &hCtrlNet);
804 if (status != ERROR_SUCCESS)
805 return setError (E_FAIL, tr("Could not open registry key \"%s\""), NetworkKey);
806
807 for (int i = 0;; ++ i)
808 {
809 char szNetworkGUID [256];
810 HKEY hConnection;
811 char szNetworkConnection [256];
812
813 len = sizeof (szNetworkGUID);
814 status = RegEnumKeyExA (hCtrlNet, i, szNetworkGUID, &len, NULL, NULL, NULL, NULL);
815 if (status != ERROR_SUCCESS)
816 break;
817
818 if (!IsTAPDevice(szNetworkGUID))
819 continue;
820
821 RTStrPrintf (szNetworkConnection, sizeof (szNetworkConnection),
822 "%s\\Connection", szNetworkGUID);
823 status = RegOpenKeyExA (hCtrlNet, szNetworkConnection, 0, KEY_READ, &hConnection);
824 if (status == ERROR_SUCCESS)
825 {
826 DWORD dwKeyType;
827 status = RegQueryValueExW (hConnection, TEXT("Name"), NULL,
828 &dwKeyType, NULL, &len);
829 if (status == ERROR_SUCCESS && dwKeyType == REG_SZ)
830 {
831 size_t uniLen = (len + sizeof (OLECHAR) - 1) / sizeof (OLECHAR);
832 Bstr name (uniLen + 1 /* extra zero */);
833 status = RegQueryValueExW (hConnection, TEXT("Name"), NULL,
834 &dwKeyType, (LPBYTE) name.mutableRaw(), &len);
835 if (status == ERROR_SUCCESS)
836 {
837 RTLogPrintf("Connection name %ls\n", name.mutableRaw());
838 /* put a trailing zero, just in case (see MSDN) */
839 name.mutableRaw() [uniLen] = 0;
840 /* create a new object and add it to the list */
841 ComObjPtr <HostNetworkInterface> iface;
842 iface.createObject();
843 /* remove the curly bracket at the end */
844 szNetworkGUID [strlen(szNetworkGUID) - 1] = '\0';
845 if (SUCCEEDED (iface->init (name, Guid (szNetworkGUID + 1))))
846 list.push_back (iface);
847 }
848 }
849 RegCloseKey (hConnection);
850 }
851 }
852 RegCloseKey (hCtrlNet);
853# endif /* RT_OS_WINDOWS */
854
855 ComObjPtr <HostNetworkInterfaceCollection> collection;
856 collection.createObject();
857 collection->init (list);
858 collection.queryInterfaceTo (networkInterfaces);
859 return S_OK;
860
861#else
862 /* Not implemented / supported on this platform. */
863 return E_NOTIMPL;
864#endif
865}
866
867STDMETHODIMP Host::COMGETTER(USBDevices)(IHostUSBDeviceCollection **aUSBDevices)
868{
869#ifdef VBOX_WITH_USB
870 if (!aUSBDevices)
871 return E_POINTER;
872
873 AutoWriteLock alock (this);
874 CHECK_READY();
875
876 MultiResult rc = checkUSBProxyService();
877 CheckComRCReturnRC (rc);
878
879 return mUSBProxyService->getDeviceCollection (aUSBDevices);
880
881#else
882 /* Note: The GUI depends on this method returning E_NOTIMPL with no
883 * extended error info to indicate that USB is simply not available
884 * (w/o treting it as a failure), for example, as in OSE */
885 return E_NOTIMPL;
886#endif
887}
888
889STDMETHODIMP Host::COMGETTER(USBDeviceFilters) (IHostUSBDeviceFilterCollection **aUSBDeviceFilters)
890{
891#ifdef VBOX_WITH_USB
892 if (!aUSBDeviceFilters)
893 return E_POINTER;
894
895 AutoWriteLock alock (this);
896 CHECK_READY();
897
898 MultiResult rc = checkUSBProxyService();
899 CheckComRCReturnRC (rc);
900
901 ComObjPtr <HostUSBDeviceFilterCollection> collection;
902 collection.createObject();
903 collection->init (mUSBDeviceFilters);
904 collection.queryInterfaceTo (aUSBDeviceFilters);
905
906 return rc;
907#else
908 /* Note: The GUI depends on this method returning E_NOTIMPL with no
909 * extended error info to indicate that USB is simply not available
910 * (w/o treting it as a failure), for example, as in OSE */
911 return E_NOTIMPL;
912#endif
913}
914
915/**
916 * Returns the number of installed logical processors
917 *
918 * @returns COM status code
919 * @param count address of result variable
920 */
921STDMETHODIMP Host::COMGETTER(ProcessorCount)(ULONG *count)
922{
923 if (!count)
924 return E_POINTER;
925 AutoWriteLock alock (this);
926 CHECK_READY();
927 *count = RTMpGetPresentCount();
928 return S_OK;
929}
930
931/**
932 * Returns the number of online logical processors
933 *
934 * @returns COM status code
935 * @param count address of result variable
936 */
937STDMETHODIMP Host::COMGETTER(ProcessorOnlineCount)(ULONG *count)
938{
939 if (!count)
940 return E_POINTER;
941 AutoWriteLock alock (this);
942 CHECK_READY();
943 *count = RTMpGetOnlineCount();
944 return S_OK;
945}
946
947/**
948 * Returns the (approximate) maximum speed of the given host CPU in MHz
949 *
950 * @returns COM status code
951 * @param cpu id to get info for.
952 * @param speed address of result variable, speed is 0 if unknown or cpuId is invalid.
953 */
954STDMETHODIMP Host::GetProcessorSpeed(ULONG aCpuId, ULONG *speed)
955{
956 if (!speed)
957 return E_POINTER;
958 AutoWriteLock alock (this);
959 CHECK_READY();
960 *speed = RTMpGetMaxFrequency(aCpuId);
961 return S_OK;
962}
963/**
964 * Returns a description string for the host CPU
965 *
966 * @returns COM status code
967 * @param cpu id to get info for.
968 * @param description address of result variable, NULL if known or cpuId is invalid.
969 */
970STDMETHODIMP Host::GetProcessorDescription(ULONG cpuId, BSTR *description)
971{
972 if (!description)
973 return E_POINTER;
974 AutoWriteLock alock (this);
975 CHECK_READY();
976 /** @todo */
977 return E_NOTIMPL;
978}
979
980
981/**
982 * Returns the amount of installed system memory in megabytes
983 *
984 * @returns COM status code
985 * @param size address of result variable
986 */
987STDMETHODIMP Host::COMGETTER(MemorySize)(ULONG *size)
988{
989 if (!size)
990 return E_POINTER;
991 AutoWriteLock alock (this);
992 CHECK_READY();
993 /** @todo */
994 return E_NOTIMPL;
995}
996
997/**
998 * Returns the current system memory free space in megabytes
999 *
1000 * @returns COM status code
1001 * @param available address of result variable
1002 */
1003STDMETHODIMP Host::COMGETTER(MemoryAvailable)(ULONG *available)
1004{
1005 if (!available)
1006 return E_POINTER;
1007 AutoWriteLock alock (this);
1008 CHECK_READY();
1009 /** @todo */
1010 return E_NOTIMPL;
1011}
1012
1013/**
1014 * Returns the name string of the host operating system
1015 *
1016 * @returns COM status code
1017 * @param os address of result variable
1018 */
1019STDMETHODIMP Host::COMGETTER(OperatingSystem)(BSTR *os)
1020{
1021 if (!os)
1022 return E_POINTER;
1023 AutoWriteLock alock (this);
1024 CHECK_READY();
1025 /** @todo */
1026 return E_NOTIMPL;
1027}
1028
1029/**
1030 * Returns the version string of the host operating system
1031 *
1032 * @returns COM status code
1033 * @param os address of result variable
1034 */
1035STDMETHODIMP Host::COMGETTER(OSVersion)(BSTR *version)
1036{
1037 if (!version)
1038 return E_POINTER;
1039 AutoWriteLock alock (this);
1040 CHECK_READY();
1041 /** @todo */
1042 return E_NOTIMPL;
1043}
1044
1045/**
1046 * Returns the current host time in milliseconds since 1970-01-01 UTC.
1047 *
1048 * @returns COM status code
1049 * @param time address of result variable
1050 */
1051STDMETHODIMP Host::COMGETTER(UTCTime)(LONG64 *aUTCTime)
1052{
1053 if (!aUTCTime)
1054 return E_POINTER;
1055 AutoWriteLock alock (this);
1056 CHECK_READY();
1057 RTTIMESPEC now;
1058 *aUTCTime = RTTimeSpecGetMilli(RTTimeNow(&now));
1059 return S_OK;
1060}
1061
1062// IHost methods
1063////////////////////////////////////////////////////////////////////////////////
1064
1065#ifdef RT_OS_WINDOWS
1066
1067/**
1068 * Returns TRUE if the Windows version is 6.0 or greater (i.e. it's Vista and
1069 * later OSes) and it has the UAC (User Account Control) feature enabled.
1070 */
1071static BOOL IsUACEnabled()
1072{
1073 LONG rc = 0;
1074
1075 OSVERSIONINFOEX info;
1076 ZeroMemory (&info, sizeof (OSVERSIONINFOEX));
1077 info.dwOSVersionInfoSize = sizeof (OSVERSIONINFOEX);
1078 rc = GetVersionEx ((OSVERSIONINFO *) &info);
1079 AssertReturn (rc != 0, FALSE);
1080
1081 LogFlowFunc (("dwMajorVersion=%d, dwMinorVersion=%d\n",
1082 info.dwMajorVersion, info.dwMinorVersion));
1083
1084 /* we are interested only in Vista (and newer versions...). In all
1085 * earlier versions UAC is not present. */
1086 if (info.dwMajorVersion < 6)
1087 return FALSE;
1088
1089 /* the default EnableLUA value is 1 (Enabled) */
1090 DWORD dwEnableLUA = 1;
1091
1092 HKEY hKey;
1093 rc = RegOpenKeyExA (HKEY_LOCAL_MACHINE,
1094 "Software\\Microsoft\\Windows\\CurrentVersion\\Policies\\System",
1095 0, KEY_QUERY_VALUE, &hKey);
1096
1097 Assert (rc == ERROR_SUCCESS || rc == ERROR_PATH_NOT_FOUND);
1098 if (rc == ERROR_SUCCESS)
1099 {
1100
1101 DWORD cbEnableLUA = sizeof (dwEnableLUA);
1102 rc = RegQueryValueExA (hKey, "EnableLUA", NULL, NULL,
1103 (LPBYTE) &dwEnableLUA, &cbEnableLUA);
1104
1105 RegCloseKey (hKey);
1106
1107 Assert (rc == ERROR_SUCCESS || rc == ERROR_FILE_NOT_FOUND);
1108 }
1109
1110 LogFlowFunc (("rc=%d, dwEnableLUA=%d\n", rc, dwEnableLUA));
1111
1112 return dwEnableLUA == 1;
1113}
1114
1115struct NetworkInterfaceHelperClientData
1116{
1117 SVCHlpMsg::Code msgCode;
1118 /* for SVCHlpMsg::CreateHostNetworkInterface */
1119 Bstr name;
1120 ComObjPtr <HostNetworkInterface> iface;
1121 /* for SVCHlpMsg::RemoveHostNetworkInterface */
1122 Guid guid;
1123};
1124
1125STDMETHODIMP
1126Host::CreateHostNetworkInterface (INPTR BSTR aName,
1127 IHostNetworkInterface **aHostNetworkInterface,
1128 IProgress **aProgress)
1129{
1130 if (!aName)
1131 return E_INVALIDARG;
1132 if (!aHostNetworkInterface)
1133 return E_POINTER;
1134 if (!aProgress)
1135 return E_POINTER;
1136
1137 AutoWriteLock alock (this);
1138 CHECK_READY();
1139
1140 HRESULT rc = S_OK;
1141
1142 /* first check whether an interface with the given name already exists */
1143 {
1144 ComPtr <IHostNetworkInterfaceCollection> coll;
1145 rc = COMGETTER(NetworkInterfaces) (coll.asOutParam());
1146 CheckComRCReturnRC (rc);
1147 ComPtr <IHostNetworkInterface> iface;
1148 if (SUCCEEDED (coll->FindByName (aName, iface.asOutParam())))
1149 return setError (E_FAIL,
1150 tr ("Host network interface '%ls' already exists"), aName);
1151 }
1152
1153 /* create a progress object */
1154 ComObjPtr <Progress> progress;
1155 progress.createObject();
1156 rc = progress->init (mParent, static_cast <IHost *> (this),
1157 Bstr (tr ("Creating host network interface")),
1158 FALSE /* aCancelable */);
1159 CheckComRCReturnRC (rc);
1160 progress.queryInterfaceTo (aProgress);
1161
1162 /* create a new uninitialized host interface object */
1163 ComObjPtr <HostNetworkInterface> iface;
1164 iface.createObject();
1165 iface.queryInterfaceTo (aHostNetworkInterface);
1166
1167 /* create the networkInterfaceHelperClient() argument */
1168 std::auto_ptr <NetworkInterfaceHelperClientData>
1169 d (new NetworkInterfaceHelperClientData());
1170 AssertReturn (d.get(), E_OUTOFMEMORY);
1171
1172 d->msgCode = SVCHlpMsg::CreateHostNetworkInterface;
1173 d->name = aName;
1174 d->iface = iface;
1175
1176 rc = mParent->startSVCHelperClient (
1177 IsUACEnabled() == TRUE /* aPrivileged */,
1178 networkInterfaceHelperClient,
1179 static_cast <void *> (d.get()),
1180 progress);
1181
1182 if (SUCCEEDED (rc))
1183 {
1184 /* d is now owned by networkInterfaceHelperClient(), so release it */
1185 d.release();
1186 }
1187
1188 return rc;
1189}
1190
1191STDMETHODIMP
1192Host::RemoveHostNetworkInterface (INPTR GUIDPARAM aId,
1193 IHostNetworkInterface **aHostNetworkInterface,
1194 IProgress **aProgress)
1195{
1196 if (!aHostNetworkInterface)
1197 return E_POINTER;
1198 if (!aProgress)
1199 return E_POINTER;
1200
1201 AutoWriteLock alock (this);
1202 CHECK_READY();
1203
1204 HRESULT rc = S_OK;
1205
1206 /* first check whether an interface with the given name already exists */
1207 {
1208 ComPtr <IHostNetworkInterfaceCollection> coll;
1209 rc = COMGETTER(NetworkInterfaces) (coll.asOutParam());
1210 CheckComRCReturnRC (rc);
1211 ComPtr <IHostNetworkInterface> iface;
1212 if (FAILED (coll->FindById (aId, iface.asOutParam())))
1213 return setError (E_FAIL,
1214 tr ("Host network interface with UUID {%Vuuid} does not exist"),
1215 Guid (aId).raw());
1216
1217 /* return the object to be removed to the caller */
1218 iface.queryInterfaceTo (aHostNetworkInterface);
1219 }
1220
1221 /* create a progress object */
1222 ComObjPtr <Progress> progress;
1223 progress.createObject();
1224 rc = progress->init (mParent, static_cast <IHost *> (this),
1225 Bstr (tr ("Removing host network interface")),
1226 FALSE /* aCancelable */);
1227 CheckComRCReturnRC (rc);
1228 progress.queryInterfaceTo (aProgress);
1229
1230 /* create the networkInterfaceHelperClient() argument */
1231 std::auto_ptr <NetworkInterfaceHelperClientData>
1232 d (new NetworkInterfaceHelperClientData());
1233 AssertReturn (d.get(), E_OUTOFMEMORY);
1234
1235 d->msgCode = SVCHlpMsg::RemoveHostNetworkInterface;
1236 d->guid = aId;
1237
1238 rc = mParent->startSVCHelperClient (
1239 IsUACEnabled() == TRUE /* aPrivileged */,
1240 networkInterfaceHelperClient,
1241 static_cast <void *> (d.get()),
1242 progress);
1243
1244 if (SUCCEEDED (rc))
1245 {
1246 /* d is now owned by networkInterfaceHelperClient(), so release it */
1247 d.release();
1248 }
1249
1250 return rc;
1251}
1252
1253#endif /* RT_OS_WINDOWS */
1254
1255STDMETHODIMP Host::CreateUSBDeviceFilter (INPTR BSTR aName, IHostUSBDeviceFilter **aFilter)
1256{
1257#ifdef VBOX_WITH_USB
1258 if (!aFilter)
1259 return E_POINTER;
1260
1261 if (!aName || *aName == 0)
1262 return E_INVALIDARG;
1263
1264 AutoWriteLock alock (this);
1265 CHECK_READY();
1266
1267 ComObjPtr <HostUSBDeviceFilter> filter;
1268 filter.createObject();
1269 HRESULT rc = filter->init (this, aName);
1270 ComAssertComRCRet (rc, rc);
1271 rc = filter.queryInterfaceTo (aFilter);
1272 AssertComRCReturn (rc, rc);
1273 return S_OK;
1274#else
1275 /* Note: The GUI depends on this method returning E_NOTIMPL with no
1276 * extended error info to indicate that USB is simply not available
1277 * (w/o treting it as a failure), for example, as in OSE */
1278 return E_NOTIMPL;
1279#endif
1280}
1281
1282STDMETHODIMP Host::InsertUSBDeviceFilter (ULONG aPosition, IHostUSBDeviceFilter *aFilter)
1283{
1284#ifdef VBOX_WITH_USB
1285 if (!aFilter)
1286 return E_INVALIDARG;
1287
1288 /* Note: HostUSBDeviceFilter and USBProxyService also uses this lock. */
1289 AutoWriteLock alock (this);
1290 CHECK_READY();
1291
1292 MultiResult rc = checkUSBProxyService();
1293 CheckComRCReturnRC (rc);
1294
1295 ComObjPtr <HostUSBDeviceFilter> filter = getDependentChild (aFilter);
1296 if (!filter)
1297 return setError (E_INVALIDARG,
1298 tr ("The given USB device filter is not created within "
1299 "this VirtualBox instance"));
1300
1301 if (filter->mInList)
1302 return setError (E_INVALIDARG,
1303 tr ("The given USB device filter is already in the list"));
1304
1305 /* iterate to the position... */
1306 USBDeviceFilterList::iterator it = mUSBDeviceFilters.begin();
1307 std::advance (it, aPosition);
1308 /* ...and insert */
1309 mUSBDeviceFilters.insert (it, filter);
1310 filter->mInList = true;
1311
1312 /* notify the proxy (only when the filter is active) */
1313 if (mUSBProxyService->isActive() && filter->data().mActive)
1314 {
1315 ComAssertRet (filter->id() == NULL, E_FAIL);
1316 filter->id() = mUSBProxyService->insertFilter (&filter->data().mUSBFilter);
1317 }
1318
1319 /* save the global settings */
1320 alock.unlock();
1321 return rc = mParent->saveSettings();
1322#else
1323 /* Note: The GUI depends on this method returning E_NOTIMPL with no
1324 * extended error info to indicate that USB is simply not available
1325 * (w/o treting it as a failure), for example, as in OSE */
1326 return E_NOTIMPL;
1327#endif
1328}
1329
1330STDMETHODIMP Host::RemoveUSBDeviceFilter (ULONG aPosition, IHostUSBDeviceFilter **aFilter)
1331{
1332#ifdef VBOX_WITH_USB
1333 if (!aFilter)
1334 return E_POINTER;
1335
1336 /* Note: HostUSBDeviceFilter and USBProxyService also uses this lock. */
1337 AutoWriteLock alock (this);
1338 CHECK_READY();
1339
1340 MultiResult rc = checkUSBProxyService();
1341 CheckComRCReturnRC (rc);
1342
1343 if (!mUSBDeviceFilters.size())
1344 return setError (E_INVALIDARG,
1345 tr ("The USB device filter list is empty"));
1346
1347 if (aPosition >= mUSBDeviceFilters.size())
1348 return setError (E_INVALIDARG,
1349 tr ("Invalid position: %lu (must be in range [0, %lu])"),
1350 aPosition, mUSBDeviceFilters.size() - 1);
1351
1352 ComObjPtr <HostUSBDeviceFilter> filter;
1353 {
1354 /* iterate to the position... */
1355 USBDeviceFilterList::iterator it = mUSBDeviceFilters.begin();
1356 std::advance (it, aPosition);
1357 /* ...get an element from there... */
1358 filter = *it;
1359 /* ...and remove */
1360 filter->mInList = false;
1361 mUSBDeviceFilters.erase (it);
1362 }
1363
1364 filter.queryInterfaceTo (aFilter);
1365
1366 /* notify the proxy (only when the filter is active) */
1367 if (mUSBProxyService->isActive() && filter->data().mActive)
1368 {
1369 ComAssertRet (filter->id() != NULL, E_FAIL);
1370 mUSBProxyService->removeFilter (filter->id());
1371 filter->id() = NULL;
1372 }
1373
1374 /* save the global settings */
1375 alock.unlock();
1376 return rc = mParent->saveSettings();
1377#else
1378 /* Note: The GUI depends on this method returning E_NOTIMPL with no
1379 * extended error info to indicate that USB is simply not available
1380 * (w/o treting it as a failure), for example, as in OSE */
1381 return E_NOTIMPL;
1382#endif
1383}
1384
1385// public methods only for internal purposes
1386////////////////////////////////////////////////////////////////////////////////
1387
1388HRESULT Host::loadSettings (const settings::Key &aGlobal)
1389{
1390 using namespace settings;
1391
1392 AutoWriteLock alock (this);
1393 CHECK_READY();
1394
1395 AssertReturn (!aGlobal.isNull(), E_FAIL);
1396
1397 HRESULT rc = S_OK;
1398
1399#ifdef VBOX_WITH_USB
1400 Key::List filters = aGlobal.key ("USBDeviceFilters").keys ("DeviceFilter");
1401 for (Key::List::const_iterator it = filters.begin();
1402 it != filters.end(); ++ it)
1403 {
1404 Bstr name = (*it).stringValue ("name");
1405 bool active = (*it).value <bool> ("active");
1406
1407 Bstr vendorId = (*it).stringValue ("vendorId");
1408 Bstr productId = (*it).stringValue ("productId");
1409 Bstr revision = (*it).stringValue ("revision");
1410 Bstr manufacturer = (*it).stringValue ("manufacturer");
1411 Bstr product = (*it).stringValue ("product");
1412 Bstr serialNumber = (*it).stringValue ("serialNumber");
1413 Bstr port = (*it).stringValue ("port");
1414
1415 USBDeviceFilterAction_T action;
1416 action = USBDeviceFilterAction_Ignore;
1417 const char *actionStr = (*it).stringValue ("action");
1418 if (strcmp (actionStr, "Ignore") == 0)
1419 action = USBDeviceFilterAction_Ignore;
1420 else
1421 if (strcmp (actionStr, "Hold") == 0)
1422 action = USBDeviceFilterAction_Hold;
1423 else
1424 AssertMsgFailed (("Invalid action: '%s'\n", actionStr));
1425
1426 ComObjPtr <HostUSBDeviceFilter> filterObj;
1427 filterObj.createObject();
1428 rc = filterObj->init (this,
1429 name, active, vendorId, productId, revision,
1430 manufacturer, product, serialNumber, port,
1431 action);
1432 /* error info is set by init() when appropriate */
1433 CheckComRCBreakRC (rc);
1434
1435 mUSBDeviceFilters.push_back (filterObj);
1436 filterObj->mInList = true;
1437
1438 /* notify the proxy (only when the filter is active) */
1439 if (filterObj->data().mActive)
1440 {
1441 HostUSBDeviceFilter *flt = filterObj; /* resolve ambiguity */
1442 flt->id() = mUSBProxyService->insertFilter (&filterObj->data().mUSBFilter);
1443 }
1444 }
1445#endif /* VBOX_WITH_USB */
1446
1447 return rc;
1448}
1449
1450HRESULT Host::saveSettings (settings::Key &aGlobal)
1451{
1452 using namespace settings;
1453
1454 AutoWriteLock alock (this);
1455 CHECK_READY();
1456
1457 ComAssertRet (!aGlobal.isNull(), E_FAIL);
1458
1459#ifdef VBOX_WITH_USB
1460 /* first, delete the entry */
1461 Key filters = aGlobal.findKey ("USBDeviceFilters");
1462 if (!filters.isNull())
1463 filters.zap();
1464 /* then, recreate it */
1465 filters = aGlobal.createKey ("USBDeviceFilters");
1466
1467 USBDeviceFilterList::const_iterator it = mUSBDeviceFilters.begin();
1468 while (it != mUSBDeviceFilters.end())
1469 {
1470 AutoWriteLock filterLock (*it);
1471 const HostUSBDeviceFilter::Data &data = (*it)->data();
1472
1473 Key filter = filters.appendKey ("DeviceFilter");
1474
1475 filter.setValue <Bstr> ("name", data.mName);
1476 filter.setValue <bool> ("active", !!data.mActive);
1477
1478 /* all are optional */
1479 Bstr str;
1480 (*it)->COMGETTER (VendorId) (str.asOutParam());
1481 if (!str.isNull())
1482 filter.setValue <Bstr> ("vendorId", str);
1483
1484 (*it)->COMGETTER (ProductId) (str.asOutParam());
1485 if (!str.isNull())
1486 filter.setValue <Bstr> ("productId", str);
1487
1488 (*it)->COMGETTER (Revision) (str.asOutParam());
1489 if (!str.isNull())
1490 filter.setValue <Bstr> ("revision", str);
1491
1492 (*it)->COMGETTER (Manufacturer) (str.asOutParam());
1493 if (!str.isNull())
1494 filter.setValue <Bstr> ("manufacturer", str);
1495
1496 (*it)->COMGETTER (Product) (str.asOutParam());
1497 if (!str.isNull())
1498 filter.setValue <Bstr> ("product", str);
1499
1500 (*it)->COMGETTER (SerialNumber) (str.asOutParam());
1501 if (!str.isNull())
1502 filter.setValue <Bstr> ("serialNumber", str);
1503
1504 (*it)->COMGETTER (Port) (str.asOutParam());
1505 if (!str.isNull())
1506 filter.setValue <Bstr> ("port", str);
1507
1508 /* action is mandatory */
1509 USBDeviceFilterAction_T action = USBDeviceFilterAction_Null;
1510 (*it)->COMGETTER (Action) (&action);
1511 if (action == USBDeviceFilterAction_Ignore)
1512 filter.setStringValue ("action", "Ignore");
1513 else if (action == USBDeviceFilterAction_Hold)
1514 filter.setStringValue ("action", "Hold");
1515 else
1516 AssertMsgFailed (("Invalid action: %d\n", action));
1517
1518 ++ it;
1519 }
1520#endif /* VBOX_WITH_USB */
1521
1522 return S_OK;
1523}
1524
1525#ifdef VBOX_WITH_USB
1526/**
1527 * Called by setter methods of all USB device filters.
1528 */
1529HRESULT Host::onUSBDeviceFilterChange (HostUSBDeviceFilter *aFilter,
1530 BOOL aActiveChanged /* = FALSE */)
1531{
1532 AutoWriteLock alock (this);
1533 CHECK_READY();
1534
1535 if (aFilter->mInList)
1536 {
1537 if (aActiveChanged)
1538 {
1539 // insert/remove the filter from the proxy
1540 if (aFilter->data().mActive)
1541 {
1542 ComAssertRet (aFilter->id() == NULL, E_FAIL);
1543 aFilter->id() = mUSBProxyService->insertFilter (&aFilter->data().mUSBFilter);
1544 }
1545 else
1546 {
1547 ComAssertRet (aFilter->id() != NULL, E_FAIL);
1548 mUSBProxyService->removeFilter (aFilter->id());
1549 aFilter->id() = NULL;
1550 }
1551 }
1552 else
1553 {
1554 if (aFilter->data().mActive)
1555 {
1556 // update the filter in the proxy
1557 ComAssertRet (aFilter->id() != NULL, E_FAIL);
1558 mUSBProxyService->removeFilter (aFilter->id());
1559 aFilter->id() = mUSBProxyService->insertFilter (&aFilter->data().mUSBFilter);
1560 }
1561 }
1562
1563 // save the global settings... yeah, on every single filter property change
1564 alock.unlock();
1565 return mParent->saveSettings();
1566 }
1567
1568 return S_OK;
1569}
1570
1571
1572/**
1573 * Interface for obtaining a copy of the USBDeviceFilterList,
1574 * used by the USBProxyService.
1575 *
1576 * @param aGlobalFilters Where to put the global filter list copy.
1577 * @param aMachines Where to put the machine vector.
1578 */
1579void Host::getUSBFilters(Host::USBDeviceFilterList *aGlobalFilters, VirtualBox::SessionMachineVector *aMachines)
1580{
1581 AutoWriteLock alock (this);
1582
1583 mParent->getOpenedMachines (*aMachines);
1584 *aGlobalFilters = mUSBDeviceFilters;
1585}
1586
1587#endif /* VBOX_WITH_USB */
1588
1589// private methods
1590////////////////////////////////////////////////////////////////////////////////
1591
1592#if defined(RT_OS_LINUX) || defined(RT_OS_SOLARIS)
1593# ifdef VBOX_USE_LIBHAL
1594/**
1595 * Helper function to query the hal subsystem for information about DVD drives attached to the
1596 * system.
1597 *
1598 * @returns true if information was successfully obtained, false otherwise
1599 * @retval list drives found will be attached to this list
1600 */
1601bool Host::getDVDInfoFromHal(std::list <ComObjPtr <HostDVDDrive> > &list)
1602{
1603 bool halSuccess = false;
1604 DBusError dbusError;
1605 if (!gLibHalCheckPresence())
1606 return false;
1607 gDBusErrorInit (&dbusError);
1608 DBusConnection *dbusConnection = gDBusBusGet(DBUS_BUS_SYSTEM, &dbusError);
1609 if (dbusConnection != 0)
1610 {
1611 LibHalContext *halContext = gLibHalCtxNew();
1612 if (halContext != 0)
1613 {
1614 if (gLibHalCtxSetDBusConnection (halContext, dbusConnection))
1615 {
1616 if (gLibHalCtxInit(halContext, &dbusError))
1617 {
1618 int numDevices;
1619 char **halDevices = gLibHalFindDeviceStringMatch(halContext,
1620 "storage.drive_type", "cdrom",
1621 &numDevices, &dbusError);
1622 if (halDevices != 0)
1623 {
1624 /* Hal is installed and working, so if no devices are reported, assume
1625 that there are none. */
1626 halSuccess = true;
1627 for (int i = 0; i < numDevices; i++)
1628 {
1629 char *devNode = gLibHalDeviceGetPropertyString(halContext,
1630 halDevices[i], "block.device", &dbusError);
1631#ifdef RT_OS_SOLARIS
1632 /* The CD/DVD ioctls work only for raw device nodes. */
1633 char *tmp = getfullrawname(devNode);
1634 gLibHalFreeString(devNode);
1635 devNode = tmp;
1636#endif
1637 if (devNode != 0)
1638 {
1639// if (validateDevice(devNode, true))
1640// {
1641 Utf8Str description;
1642 char *vendor, *product;
1643 /* We do not check the error here, as this field may
1644 not even exist. */
1645 vendor = gLibHalDeviceGetPropertyString(halContext,
1646 halDevices[i], "info.vendor", 0);
1647 product = gLibHalDeviceGetPropertyString(halContext,
1648 halDevices[i], "info.product", &dbusError);
1649 if ((product != 0 && product[0] != 0))
1650 {
1651 if ((vendor != 0) && (vendor[0] != 0))
1652 {
1653 description = Utf8StrFmt ("%s %s",
1654 vendor, product);
1655 }
1656 else
1657 {
1658 description = product;
1659 }
1660 ComObjPtr <HostDVDDrive> hostDVDDriveObj;
1661 hostDVDDriveObj.createObject();
1662 hostDVDDriveObj->init (Bstr (devNode),
1663 Bstr (halDevices[i]),
1664 Bstr (description));
1665 list.push_back (hostDVDDriveObj);
1666 }
1667 else
1668 {
1669 if (product == 0)
1670 {
1671 LogRel(("Host::COMGETTER(DVDDrives): failed to get property \"info.product\" for device %s. dbus error: %s (%s)\n",
1672 halDevices[i], dbusError.name, dbusError.message));
1673 gDBusErrorFree(&dbusError);
1674 }
1675 ComObjPtr <HostDVDDrive> hostDVDDriveObj;
1676 hostDVDDriveObj.createObject();
1677 hostDVDDriveObj->init (Bstr (devNode),
1678 Bstr (halDevices[i]));
1679 list.push_back (hostDVDDriveObj);
1680 }
1681 if (vendor != 0)
1682 {
1683 gLibHalFreeString(vendor);
1684 }
1685 if (product != 0)
1686 {
1687 gLibHalFreeString(product);
1688 }
1689// }
1690// else
1691// {
1692// LogRel(("Host::COMGETTER(DVDDrives): failed to validate the block device %s as a DVD drive\n"));
1693// }
1694#ifndef RT_OS_SOLARIS
1695 gLibHalFreeString(devNode);
1696#else
1697 free(devNode);
1698#endif
1699 }
1700 else
1701 {
1702 LogRel(("Host::COMGETTER(DVDDrives): failed to get property \"block.device\" for device %s. dbus error: %s (%s)\n",
1703 halDevices[i], dbusError.name, dbusError.message));
1704 gDBusErrorFree(&dbusError);
1705 }
1706 }
1707 gLibHalFreeStringArray(halDevices);
1708 }
1709 else
1710 {
1711 LogRel(("Host::COMGETTER(DVDDrives): failed to get devices with capability \"storage.cdrom\". dbus error: %s (%s)\n", dbusError.name, dbusError.message));
1712 gDBusErrorFree(&dbusError);
1713 }
1714 if (!gLibHalCtxShutdown(halContext, &dbusError)) /* what now? */
1715 {
1716 LogRel(("Host::COMGETTER(DVDDrives): failed to shutdown the libhal context. dbus error: %s (%s)\n", dbusError.name, dbusError.message));
1717 gDBusErrorFree(&dbusError);
1718 }
1719 }
1720 else
1721 {
1722 LogRel(("Host::COMGETTER(DVDDrives): failed to initialise libhal context. dbus error: %s (%s)\n", dbusError.name, dbusError.message));
1723 gDBusErrorFree(&dbusError);
1724 }
1725 gLibHalCtxFree(halContext);
1726 }
1727 else
1728 {
1729 LogRel(("Host::COMGETTER(DVDDrives): failed to set libhal connection to dbus.\n"));
1730 }
1731 }
1732 else
1733 {
1734 LogRel(("Host::COMGETTER(DVDDrives): failed to get a libhal context - out of memory?\n"));
1735 }
1736 gDBusConnectionUnref(dbusConnection);
1737 }
1738 else
1739 {
1740 LogRel(("Host::COMGETTER(DVDDrives): failed to connect to dbus. dbus error: %s (%s)\n", dbusError.name, dbusError.message));
1741 gDBusErrorFree(&dbusError);
1742 }
1743 return halSuccess;
1744}
1745
1746
1747/**
1748 * Helper function to query the hal subsystem for information about floppy drives attached to the
1749 * system.
1750 *
1751 * @returns true if information was successfully obtained, false otherwise
1752 * @retval list drives found will be attached to this list
1753 */
1754bool Host::getFloppyInfoFromHal(std::list <ComObjPtr <HostFloppyDrive> > &list)
1755{
1756 bool halSuccess = false;
1757 DBusError dbusError;
1758 if (!gLibHalCheckPresence())
1759 return false;
1760 gDBusErrorInit (&dbusError);
1761 DBusConnection *dbusConnection = gDBusBusGet(DBUS_BUS_SYSTEM, &dbusError);
1762 if (dbusConnection != 0)
1763 {
1764 LibHalContext *halContext = gLibHalCtxNew();
1765 if (halContext != 0)
1766 {
1767 if (gLibHalCtxSetDBusConnection (halContext, dbusConnection))
1768 {
1769 if (gLibHalCtxInit(halContext, &dbusError))
1770 {
1771 int numDevices;
1772 char **halDevices = gLibHalFindDeviceStringMatch(halContext,
1773 "storage.drive_type", "floppy",
1774 &numDevices, &dbusError);
1775 if (halDevices != 0)
1776 {
1777 /* Hal is installed and working, so if no devices are reported, assume
1778 that there are none. */
1779 halSuccess = true;
1780 for (int i = 0; i < numDevices; i++)
1781 {
1782 char *driveType = gLibHalDeviceGetPropertyString(halContext,
1783 halDevices[i], "storage.drive_type", 0);
1784 if (driveType != 0)
1785 {
1786 if (strcmp(driveType, "floppy") != 0)
1787 {
1788 gLibHalFreeString(driveType);
1789 continue;
1790 }
1791 gLibHalFreeString(driveType);
1792 }
1793 else
1794 {
1795 /* An error occurred. The attribute "storage.drive_type"
1796 probably didn't exist. */
1797 continue;
1798 }
1799 char *devNode = gLibHalDeviceGetPropertyString(halContext,
1800 halDevices[i], "block.device", &dbusError);
1801 if (devNode != 0)
1802 {
1803// if (validateDevice(devNode, false))
1804// {
1805 Utf8Str description;
1806 char *vendor, *product;
1807 /* We do not check the error here, as this field may
1808 not even exist. */
1809 vendor = gLibHalDeviceGetPropertyString(halContext,
1810 halDevices[i], "info.vendor", 0);
1811 product = gLibHalDeviceGetPropertyString(halContext,
1812 halDevices[i], "info.product", &dbusError);
1813 if ((product != 0) && (product[0] != 0))
1814 {
1815 if ((vendor != 0) && (vendor[0] != 0))
1816 {
1817 description = Utf8StrFmt ("%s %s",
1818 vendor, product);
1819 }
1820 else
1821 {
1822 description = product;
1823 }
1824 ComObjPtr <HostFloppyDrive> hostFloppyDrive;
1825 hostFloppyDrive.createObject();
1826 hostFloppyDrive->init (Bstr (devNode),
1827 Bstr (halDevices[i]),
1828 Bstr (description));
1829 list.push_back (hostFloppyDrive);
1830 }
1831 else
1832 {
1833 if (product == 0)
1834 {
1835 LogRel(("Host::COMGETTER(FloppyDrives): failed to get property \"info.product\" for device %s. dbus error: %s (%s)\n",
1836 halDevices[i], dbusError.name, dbusError.message));
1837 gDBusErrorFree(&dbusError);
1838 }
1839 ComObjPtr <HostFloppyDrive> hostFloppyDrive;
1840 hostFloppyDrive.createObject();
1841 hostFloppyDrive->init (Bstr (devNode),
1842 Bstr (halDevices[i]));
1843 list.push_back (hostFloppyDrive);
1844 }
1845 if (vendor != 0)
1846 {
1847 gLibHalFreeString(vendor);
1848 }
1849 if (product != 0)
1850 {
1851 gLibHalFreeString(product);
1852 }
1853// }
1854// else
1855// {
1856// LogRel(("Host::COMGETTER(FloppyDrives): failed to validate the block device %s as a floppy drive\n"));
1857// }
1858 gLibHalFreeString(devNode);
1859 }
1860 else
1861 {
1862 LogRel(("Host::COMGETTER(FloppyDrives): failed to get property \"block.device\" for device %s. dbus error: %s (%s)\n",
1863 halDevices[i], dbusError.name, dbusError.message));
1864 gDBusErrorFree(&dbusError);
1865 }
1866 }
1867 gLibHalFreeStringArray(halDevices);
1868 }
1869 else
1870 {
1871 LogRel(("Host::COMGETTER(FloppyDrives): failed to get devices with capability \"storage.cdrom\". dbus error: %s (%s)\n", dbusError.name, dbusError.message));
1872 gDBusErrorFree(&dbusError);
1873 }
1874 if (!gLibHalCtxShutdown(halContext, &dbusError)) /* what now? */
1875 {
1876 LogRel(("Host::COMGETTER(FloppyDrives): failed to shutdown the libhal context. dbus error: %s (%s)\n", dbusError.name, dbusError.message));
1877 gDBusErrorFree(&dbusError);
1878 }
1879 }
1880 else
1881 {
1882 LogRel(("Host::COMGETTER(FloppyDrives): failed to initialise libhal context. dbus error: %s (%s)\n", dbusError.name, dbusError.message));
1883 gDBusErrorFree(&dbusError);
1884 }
1885 gLibHalCtxFree(halContext);
1886 }
1887 else
1888 {
1889 LogRel(("Host::COMGETTER(FloppyDrives): failed to set libhal connection to dbus.\n"));
1890 }
1891 }
1892 else
1893 {
1894 LogRel(("Host::COMGETTER(FloppyDrives): failed to get a libhal context - out of memory?\n"));
1895 }
1896 gDBusConnectionUnref(dbusConnection);
1897 }
1898 else
1899 {
1900 LogRel(("Host::COMGETTER(FloppyDrives): failed to connect to dbus. dbus error: %s (%s)\n", dbusError.name, dbusError.message));
1901 gDBusErrorFree(&dbusError);
1902 }
1903 return halSuccess;
1904}
1905# endif /* VBOX_USE_HAL defined */
1906
1907/**
1908 * Helper function to parse the given mount file and add found entries
1909 */
1910void Host::parseMountTable(char *mountTable, std::list <ComObjPtr <HostDVDDrive> > &list)
1911{
1912#ifdef RT_OS_LINUX
1913 FILE *mtab = setmntent(mountTable, "r");
1914 if (mtab)
1915 {
1916 struct mntent *mntent;
1917 char *mnt_type;
1918 char *mnt_dev;
1919 char *tmp;
1920 while ((mntent = getmntent(mtab)))
1921 {
1922 mnt_type = (char*)malloc(strlen(mntent->mnt_type) + 1);
1923 mnt_dev = (char*)malloc(strlen(mntent->mnt_fsname) + 1);
1924 strcpy(mnt_type, mntent->mnt_type);
1925 strcpy(mnt_dev, mntent->mnt_fsname);
1926 // supermount fs case
1927 if (strcmp(mnt_type, "supermount") == 0)
1928 {
1929 tmp = strstr(mntent->mnt_opts, "fs=");
1930 if (tmp)
1931 {
1932 free(mnt_type);
1933 mnt_type = strdup(tmp + strlen("fs="));
1934 if (mnt_type)
1935 {
1936 tmp = strchr(mnt_type, ',');
1937 if (tmp)
1938 *tmp = '\0';
1939 }
1940 }
1941 tmp = strstr(mntent->mnt_opts, "dev=");
1942 if (tmp)
1943 {
1944 free(mnt_dev);
1945 mnt_dev = strdup(tmp + strlen("dev="));
1946 if (mnt_dev)
1947 {
1948 tmp = strchr(mnt_dev, ',');
1949 if (tmp)
1950 *tmp = '\0';
1951 }
1952 }
1953 }
1954 // use strstr here to cover things fs types like "udf,iso9660"
1955 if (strstr(mnt_type, "iso9660") == 0)
1956 {
1957 /** @todo check whether we've already got the drive in our list! */
1958 if (validateDevice(mnt_dev, true))
1959 {
1960 ComObjPtr <HostDVDDrive> hostDVDDriveObj;
1961 hostDVDDriveObj.createObject();
1962 hostDVDDriveObj->init (Bstr (mnt_dev));
1963 list.push_back (hostDVDDriveObj);
1964 }
1965 }
1966 free(mnt_dev);
1967 free(mnt_type);
1968 }
1969 endmntent(mtab);
1970 }
1971#else // RT_OS_SOLARIS
1972 FILE *mntFile = fopen(mountTable, "r");
1973 if (mntFile)
1974 {
1975 struct mnttab mntTab;
1976 while (getmntent(mntFile, &mntTab) == 0)
1977 {
1978 char *mountName = strdup(mntTab.mnt_special);
1979 char *mountPoint = strdup(mntTab.mnt_mountp);
1980 char *mountFSType = strdup(mntTab.mnt_fstype);
1981
1982 // skip devices we are not interested in
1983 if ((*mountName && mountName[0] == '/') && // skip 'fake' devices (like -hosts, proc, fd, swap)
1984 (*mountFSType && (strcmp(mountFSType, "devfs") != 0 && // skip devfs (i.e. /devices)
1985 strcmp(mountFSType, "dev") != 0 && // skip dev (i.e. /dev)
1986 strcmp(mountFSType, "lofs") != 0)) && // skip loop-back file-system (lofs)
1987 (*mountPoint && strcmp(mountPoint, "/") != 0)) // skip point '/' (Can CD/DVD be mounted at '/' ???)
1988 {
1989 char *rawDevName = getfullrawname(mountName);
1990 if (validateDevice(rawDevName, true))
1991 {
1992 ComObjPtr <HostDVDDrive> hostDVDDriveObj;
1993 hostDVDDriveObj.createObject();
1994 hostDVDDriveObj->init (Bstr (rawDevName));
1995 list.push_back (hostDVDDriveObj);
1996 }
1997 free(rawDevName);
1998 }
1999
2000 free(mountName);
2001 free(mountPoint);
2002 free(mountFSType);
2003 }
2004
2005 fclose(mntFile);
2006 }
2007#endif
2008}
2009
2010/**
2011 * Helper function to check whether the given device node is a valid drive
2012 */
2013bool Host::validateDevice(const char *deviceNode, bool isCDROM)
2014{
2015 struct stat statInfo;
2016 bool retValue = false;
2017
2018 // sanity check
2019 if (!deviceNode)
2020 {
2021 return false;
2022 }
2023
2024 // first a simple stat() call
2025 if (stat(deviceNode, &statInfo) < 0)
2026 {
2027 return false;
2028 } else
2029 {
2030 if (isCDROM)
2031 {
2032 if (S_ISCHR(statInfo.st_mode) || S_ISBLK(statInfo.st_mode))
2033 {
2034 int fileHandle;
2035 // now try to open the device
2036 fileHandle = open(deviceNode, O_RDONLY | O_NONBLOCK, 0);
2037 if (fileHandle >= 0)
2038 {
2039 cdrom_subchnl cdChannelInfo;
2040 cdChannelInfo.cdsc_format = CDROM_MSF;
2041 // this call will finally reveal the whole truth
2042#ifdef RT_OS_LINUX
2043 if ((ioctl(fileHandle, CDROMSUBCHNL, &cdChannelInfo) == 0) ||
2044 (errno == EIO) || (errno == ENOENT) ||
2045 (errno == EINVAL) || (errno == ENOMEDIUM))
2046#else
2047 if ((ioctl(fileHandle, CDROMSUBCHNL, &cdChannelInfo) == 0) ||
2048 (errno == EIO) || (errno == ENOENT) ||
2049 (errno == EINVAL))
2050#endif
2051 {
2052 retValue = true;
2053 }
2054 close(fileHandle);
2055 }
2056 }
2057 } else
2058 {
2059 // floppy case
2060 if (S_ISCHR(statInfo.st_mode) || S_ISBLK(statInfo.st_mode))
2061 {
2062 /// @todo do some more testing, maybe a nice IOCTL!
2063 retValue = true;
2064 }
2065 }
2066 }
2067 return retValue;
2068}
2069#endif // RT_OS_LINUX || RT_OS_SOLARIS
2070
2071#ifdef VBOX_WITH_USB
2072/**
2073 * Checks for the presense and status of the USB Proxy Service.
2074 * Returns S_OK when the Proxy is present and OK, or E_FAIL and a
2075 * corresponding error message otherwise. Intended to be used by methods
2076 * that rely on the Proxy Service availability.
2077 *
2078 * @note This method may return a warning result code. It is recommended to use
2079 * MultiError to store the return value.
2080 *
2081 * @note Locks this object for reading.
2082 */
2083HRESULT Host::checkUSBProxyService()
2084{
2085 AutoWriteLock alock (this);
2086 CHECK_READY();
2087
2088 AssertReturn (mUSBProxyService, E_FAIL);
2089 if (!mUSBProxyService->isActive())
2090 {
2091 /* disable the USB controller completely to avoid assertions if the
2092 * USB proxy service could not start. */
2093
2094 if (mUSBProxyService->getLastError() == VERR_FILE_NOT_FOUND)
2095 return setWarning (E_FAIL,
2096 tr ("Could not load the Host USB Proxy Service (%Vrc). "
2097 "The service might be not installed on the host computer"),
2098 mUSBProxyService->getLastError());
2099 if (mUSBProxyService->getLastError() == VINF_SUCCESS)
2100 return setWarning (E_FAIL,
2101 tr ("The USB Proxy Service has not yet been ported to this host"));
2102 return setWarning (E_FAIL,
2103 tr ("Could not load the Host USB Proxy service (%Vrc)"),
2104 mUSBProxyService->getLastError());
2105 }
2106
2107 return S_OK;
2108}
2109#endif /* VBOX_WITH_USB */
2110
2111#ifdef RT_OS_WINDOWS
2112
2113/* The original source of the VBoxTAP adapter creation/destruction code has the following copyright */
2114/*
2115 Copyright 2004 by the Massachusetts Institute of Technology
2116
2117 All rights reserved.
2118
2119 Permission to use, copy, modify, and distribute this software and its
2120 documentation for any purpose and without fee is hereby granted,
2121 provided that the above copyright notice appear in all copies and that
2122 both that copyright notice and this permission notice appear in
2123 supporting documentation, and that the name of the Massachusetts
2124 Institute of Technology (M.I.T.) not be used in advertising or publicity
2125 pertaining to distribution of the software without specific, written
2126 prior permission.
2127
2128 M.I.T. DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
2129 ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
2130 M.I.T. BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
2131 ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
2132 WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
2133 ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
2134 SOFTWARE.
2135*/
2136
2137
2138#define NETSHELL_LIBRARY _T("netshell.dll")
2139
2140/**
2141 * Use the IShellFolder API to rename the connection.
2142 */
2143static HRESULT rename_shellfolder (PCWSTR wGuid, PCWSTR wNewName)
2144{
2145 /* This is the GUID for the network connections folder. It is constant.
2146 * {7007ACC7-3202-11D1-AAD2-00805FC1270E} */
2147 const GUID CLSID_NetworkConnections = {
2148 0x7007ACC7, 0x3202, 0x11D1, {
2149 0xAA, 0xD2, 0x00, 0x80, 0x5F, 0xC1, 0x27, 0x0E
2150 }
2151 };
2152
2153 LPITEMIDLIST pidl = NULL;
2154 IShellFolder *pShellFolder = NULL;
2155 HRESULT hr;
2156
2157 /* Build the display name in the form "::{GUID}". */
2158 if (wcslen (wGuid) >= MAX_PATH)
2159 return E_INVALIDARG;
2160 WCHAR szAdapterGuid[MAX_PATH + 2] = {0};
2161 swprintf (szAdapterGuid, L"::%ls", wGuid);
2162
2163 /* Create an instance of the network connections folder. */
2164 hr = CoCreateInstance (CLSID_NetworkConnections, NULL,
2165 CLSCTX_INPROC_SERVER, IID_IShellFolder,
2166 reinterpret_cast <LPVOID *> (&pShellFolder));
2167 /* Parse the display name. */
2168 if (SUCCEEDED (hr))
2169 {
2170 hr = pShellFolder->ParseDisplayName (NULL, NULL, szAdapterGuid, NULL,
2171 &pidl, NULL);
2172 }
2173 if (SUCCEEDED (hr))
2174 {
2175 hr = pShellFolder->SetNameOf (NULL, pidl, wNewName, SHGDN_NORMAL,
2176 &pidl);
2177 }
2178
2179 CoTaskMemFree (pidl);
2180
2181 if (pShellFolder)
2182 pShellFolder->Release();
2183
2184 return hr;
2185}
2186
2187extern "C" HRESULT RenameConnection (PCWSTR GuidString, PCWSTR NewName)
2188{
2189 typedef HRESULT (WINAPI *lpHrRenameConnection) (const GUID *, PCWSTR);
2190 lpHrRenameConnection RenameConnectionFunc = NULL;
2191 HRESULT status;
2192
2193 /* First try the IShellFolder interface, which was unimplemented
2194 * for the network connections folder before XP. */
2195 status = rename_shellfolder (GuidString, NewName);
2196 if (status == E_NOTIMPL)
2197 {
2198/** @todo that code doesn't seem to work! */
2199 /* The IShellFolder interface is not implemented on this platform.
2200 * Try the (undocumented) HrRenameConnection API in the netshell
2201 * library. */
2202 CLSID clsid;
2203 HINSTANCE hNetShell;
2204 status = CLSIDFromString ((LPOLESTR) GuidString, &clsid);
2205 if (FAILED(status))
2206 return E_FAIL;
2207 hNetShell = LoadLibrary (NETSHELL_LIBRARY);
2208 if (hNetShell == NULL)
2209 return E_FAIL;
2210 RenameConnectionFunc =
2211 (lpHrRenameConnection) GetProcAddress (hNetShell,
2212 "HrRenameConnection");
2213 if (RenameConnectionFunc == NULL)
2214 {
2215 FreeLibrary (hNetShell);
2216 return E_FAIL;
2217 }
2218 status = RenameConnectionFunc (&clsid, NewName);
2219 FreeLibrary (hNetShell);
2220 }
2221 if (FAILED (status))
2222 return status;
2223
2224 return S_OK;
2225}
2226
2227#define DRIVERHWID _T("vboxtap")
2228
2229#define SetErrBreak(strAndArgs) \
2230 if (1) { \
2231 aErrMsg = Utf8StrFmt strAndArgs; vrc = VERR_GENERAL_FAILURE; break; \
2232 } else do {} while (0)
2233
2234/* static */
2235int Host::createNetworkInterface (SVCHlpClient *aClient,
2236 const Utf8Str &aName,
2237 Guid &aGUID, Utf8Str &aErrMsg)
2238{
2239 LogFlowFuncEnter();
2240 LogFlowFunc (("Network connection name = '%s'\n", aName.raw()));
2241
2242 AssertReturn (aClient, VERR_INVALID_POINTER);
2243 AssertReturn (!aName.isNull(), VERR_INVALID_PARAMETER);
2244
2245 int vrc = VINF_SUCCESS;
2246
2247 HDEVINFO hDeviceInfo = INVALID_HANDLE_VALUE;
2248 SP_DEVINFO_DATA DeviceInfoData;
2249 DWORD ret = 0;
2250 BOOL found = FALSE;
2251 BOOL registered = FALSE;
2252 BOOL destroyList = FALSE;
2253 TCHAR pCfgGuidString [50];
2254
2255 do
2256 {
2257 BOOL ok;
2258 GUID netGuid;
2259 SP_DRVINFO_DATA DriverInfoData;
2260 SP_DEVINSTALL_PARAMS DeviceInstallParams;
2261 TCHAR className [MAX_PATH];
2262 DWORD index = 0;
2263 PSP_DRVINFO_DETAIL_DATA pDriverInfoDetail;
2264 /* for our purposes, 2k buffer is more
2265 * than enough to obtain the hardware ID
2266 * of the VBoxTAP driver. */
2267 DWORD detailBuf [2048];
2268
2269 HKEY hkey = NULL;
2270 DWORD cbSize;
2271 DWORD dwValueType;
2272
2273 /* initialize the structure size */
2274 DeviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);
2275 DriverInfoData.cbSize = sizeof(SP_DRVINFO_DATA);
2276
2277 /* copy the net class GUID */
2278 memcpy(&netGuid, &GUID_DEVCLASS_NET, sizeof(GUID_DEVCLASS_NET));
2279
2280 /* create an empty device info set associated with the net class GUID */
2281 hDeviceInfo = SetupDiCreateDeviceInfoList (&netGuid, NULL);
2282 if (hDeviceInfo == INVALID_HANDLE_VALUE)
2283 SetErrBreak (("SetupDiCreateDeviceInfoList failed (0x%08X)",
2284 GetLastError()));
2285
2286 /* get the class name from GUID */
2287 ok = SetupDiClassNameFromGuid (&netGuid, className, MAX_PATH, NULL);
2288 if (!ok)
2289 SetErrBreak (("SetupDiClassNameFromGuid failed (0x%08X)",
2290 GetLastError()));
2291
2292 /* create a device info element and add the new device instance
2293 * key to registry */
2294 ok = SetupDiCreateDeviceInfo (hDeviceInfo, className, &netGuid, NULL, NULL,
2295 DICD_GENERATE_ID, &DeviceInfoData);
2296 if (!ok)
2297 SetErrBreak (("SetupDiCreateDeviceInfo failed (0x%08X)",
2298 GetLastError()));
2299
2300 /* select the newly created device info to be the currently
2301 selected member */
2302 ok = SetupDiSetSelectedDevice (hDeviceInfo, &DeviceInfoData);
2303 if (!ok)
2304 SetErrBreak (("SetupDiSetSelectedDevice failed (0x%08X)",
2305 GetLastError()));
2306
2307 /* build a list of class drivers */
2308 ok = SetupDiBuildDriverInfoList (hDeviceInfo, &DeviceInfoData,
2309 SPDIT_CLASSDRIVER);
2310 if (!ok)
2311 SetErrBreak (("SetupDiBuildDriverInfoList failed (0x%08X)",
2312 GetLastError()));
2313
2314 destroyList = TRUE;
2315
2316 /* enumerate the driver info list */
2317 while (TRUE)
2318 {
2319 BOOL ret;
2320
2321 ret = SetupDiEnumDriverInfo (hDeviceInfo, &DeviceInfoData,
2322 SPDIT_CLASSDRIVER, index, &DriverInfoData);
2323
2324 /* if the function failed and GetLastError() returned
2325 * ERROR_NO_MORE_ITEMS, then we have reached the end of the
2326 * list. Othewise there was something wrong with this
2327 * particular driver. */
2328 if (!ret)
2329 {
2330 if(GetLastError() == ERROR_NO_MORE_ITEMS)
2331 break;
2332 else
2333 {
2334 index++;
2335 continue;
2336 }
2337 }
2338
2339 pDriverInfoDetail = (PSP_DRVINFO_DETAIL_DATA) detailBuf;
2340 pDriverInfoDetail->cbSize = sizeof(SP_DRVINFO_DETAIL_DATA);
2341
2342 /* if we successfully find the hardware ID and it turns out to
2343 * be the one for the loopback driver, then we are done. */
2344 if (SetupDiGetDriverInfoDetail (hDeviceInfo,
2345 &DeviceInfoData,
2346 &DriverInfoData,
2347 pDriverInfoDetail,
2348 sizeof (detailBuf),
2349 NULL))
2350 {
2351 TCHAR * t;
2352
2353 /* pDriverInfoDetail->HardwareID is a MULTISZ string. Go through the
2354 * whole list and see if there is a match somewhere. */
2355 t = pDriverInfoDetail->HardwareID;
2356 while (t && *t && t < (TCHAR *) &detailBuf [sizeof(detailBuf) / sizeof (detailBuf[0])])
2357 {
2358 if (!_tcsicmp(t, DRIVERHWID))
2359 break;
2360
2361 t += _tcslen(t) + 1;
2362 }
2363
2364 if (t && *t && t < (TCHAR *) &detailBuf [sizeof(detailBuf) / sizeof (detailBuf[0])])
2365 {
2366 found = TRUE;
2367 break;
2368 }
2369 }
2370
2371 index ++;
2372 }
2373
2374 if (!found)
2375 SetErrBreak ((tr ("Could not find Host Interface Networking driver! "
2376 "Please reinstall")));
2377
2378 /* set the loopback driver to be the currently selected */
2379 ok = SetupDiSetSelectedDriver (hDeviceInfo, &DeviceInfoData,
2380 &DriverInfoData);
2381 if (!ok)
2382 SetErrBreak (("SetupDiSetSelectedDriver failed (0x%08X)",
2383 GetLastError()));
2384
2385 /* register the phantom device to prepare for install */
2386 ok = SetupDiCallClassInstaller (DIF_REGISTERDEVICE, hDeviceInfo,
2387 &DeviceInfoData);
2388 if (!ok)
2389 SetErrBreak (("SetupDiCallClassInstaller failed (0x%08X)",
2390 GetLastError()));
2391
2392 /* registered, but remove if errors occur in the following code */
2393 registered = TRUE;
2394
2395 /* ask the installer if we can install the device */
2396 ok = SetupDiCallClassInstaller (DIF_ALLOW_INSTALL, hDeviceInfo,
2397 &DeviceInfoData);
2398 if (!ok)
2399 {
2400 if (GetLastError() != ERROR_DI_DO_DEFAULT)
2401 SetErrBreak (("SetupDiCallClassInstaller (DIF_ALLOW_INSTALL) failed (0x%08X)",
2402 GetLastError()));
2403 /* that's fine */
2404 }
2405
2406 /* install the files first */
2407 ok = SetupDiCallClassInstaller (DIF_INSTALLDEVICEFILES, hDeviceInfo,
2408 &DeviceInfoData);
2409 if (!ok)
2410 SetErrBreak (("SetupDiCallClassInstaller (DIF_INSTALLDEVICEFILES) failed (0x%08X)",
2411 GetLastError()));
2412
2413 /* get the device install parameters and disable filecopy */
2414 DeviceInstallParams.cbSize = sizeof(SP_DEVINSTALL_PARAMS);
2415 ok = SetupDiGetDeviceInstallParams (hDeviceInfo, &DeviceInfoData,
2416 &DeviceInstallParams);
2417 if (ok)
2418 {
2419 DeviceInstallParams.Flags |= DI_NOFILECOPY;
2420 ok = SetupDiSetDeviceInstallParams (hDeviceInfo, &DeviceInfoData,
2421 &DeviceInstallParams);
2422 if (!ok)
2423 SetErrBreak (("SetupDiSetDeviceInstallParams failed (0x%08X)",
2424 GetLastError()));
2425 }
2426
2427 /*
2428 * Register any device-specific co-installers for this device,
2429 */
2430
2431 ok = SetupDiCallClassInstaller (DIF_REGISTER_COINSTALLERS,
2432 hDeviceInfo,
2433 &DeviceInfoData);
2434 if (!ok)
2435 SetErrBreak (("SetupDiCallClassInstaller (DIF_REGISTER_COINSTALLERS) failed (0x%08X)",
2436 GetLastError()));
2437
2438 /*
2439 * install any installer-specified interfaces.
2440 * and then do the real install
2441 */
2442 ok = SetupDiCallClassInstaller (DIF_INSTALLINTERFACES,
2443 hDeviceInfo,
2444 &DeviceInfoData);
2445 if (!ok)
2446 SetErrBreak (("SetupDiCallClassInstaller (DIF_INSTALLINTERFACES) failed (0x%08X)",
2447 GetLastError()));
2448
2449 ok = SetupDiCallClassInstaller (DIF_INSTALLDEVICE,
2450 hDeviceInfo,
2451 &DeviceInfoData);
2452 if (!ok)
2453 SetErrBreak (("SetupDiCallClassInstaller (DIF_INSTALLDEVICE) failed (0x%08X)",
2454 GetLastError()));
2455
2456 /* Figure out NetCfgInstanceId */
2457 hkey = SetupDiOpenDevRegKey (hDeviceInfo,
2458 &DeviceInfoData,
2459 DICS_FLAG_GLOBAL,
2460 0,
2461 DIREG_DRV,
2462 KEY_READ);
2463 if (hkey == INVALID_HANDLE_VALUE)
2464 SetErrBreak (("SetupDiOpenDevRegKey failed (0x%08X)",
2465 GetLastError()));
2466
2467 cbSize = sizeof (pCfgGuidString);
2468 DWORD ret;
2469 ret = RegQueryValueEx (hkey, _T ("NetCfgInstanceId"), NULL,
2470 &dwValueType, (LPBYTE) pCfgGuidString, &cbSize);
2471 RegCloseKey (hkey);
2472
2473 ret = RenameConnection (pCfgGuidString, Bstr (aName));
2474 if (FAILED (ret))
2475 SetErrBreak (("Failed to set interface name (ret=0x%08X, "
2476 "pCfgGuidString='%ls', cbSize=%d)",
2477 ret, pCfgGuidString, cbSize));
2478 }
2479 while (0);
2480
2481 /*
2482 * cleanup
2483 */
2484
2485 if (hDeviceInfo != INVALID_HANDLE_VALUE)
2486 {
2487 /* an error has occured, but the device is registered, we must remove it */
2488 if (ret != 0 && registered)
2489 SetupDiCallClassInstaller (DIF_REMOVE, hDeviceInfo, &DeviceInfoData);
2490
2491 found = SetupDiDeleteDeviceInfo (hDeviceInfo, &DeviceInfoData);
2492
2493 /* destroy the driver info list */
2494 if (destroyList)
2495 SetupDiDestroyDriverInfoList (hDeviceInfo, &DeviceInfoData,
2496 SPDIT_CLASSDRIVER);
2497 /* clean up the device info set */
2498 SetupDiDestroyDeviceInfoList (hDeviceInfo);
2499 }
2500
2501 /* return the network connection GUID on success */
2502 if (VBOX_SUCCESS (vrc))
2503 {
2504 /* remove the curly bracket at the end */
2505 pCfgGuidString [_tcslen (pCfgGuidString) - 1] = '\0';
2506 LogFlowFunc (("Network connection GUID string = {%ls}\n", pCfgGuidString + 1));
2507
2508 aGUID = Guid (Utf8Str (pCfgGuidString + 1));
2509 LogFlowFunc (("Network connection GUID = {%Vuuid}\n", aGUID.raw()));
2510 Assert (!aGUID.isEmpty());
2511 }
2512
2513 LogFlowFunc (("vrc=%Vrc\n", vrc));
2514 LogFlowFuncLeave();
2515 return vrc;
2516}
2517
2518/* static */
2519int Host::removeNetworkInterface (SVCHlpClient *aClient,
2520 const Guid &aGUID,
2521 Utf8Str &aErrMsg)
2522{
2523 LogFlowFuncEnter();
2524 LogFlowFunc (("Network connection GUID = {%Vuuid}\n", aGUID.raw()));
2525
2526 AssertReturn (aClient, VERR_INVALID_POINTER);
2527 AssertReturn (!aGUID.isEmpty(), VERR_INVALID_PARAMETER);
2528
2529 int vrc = VINF_SUCCESS;
2530
2531 do
2532 {
2533 TCHAR lszPnPInstanceId [512] = {0};
2534
2535 /* We have to find the device instance ID through a registry search */
2536
2537 HKEY hkeyNetwork = 0;
2538 HKEY hkeyConnection = 0;
2539
2540 do
2541 {
2542 char strRegLocation [256];
2543 sprintf (strRegLocation,
2544 "SYSTEM\\CurrentControlSet\\Control\\Network\\"
2545 "{4D36E972-E325-11CE-BFC1-08002BE10318}\\{%s}",
2546 aGUID.toString().raw());
2547 LONG status;
2548 status = RegOpenKeyExA (HKEY_LOCAL_MACHINE, strRegLocation, 0,
2549 KEY_READ, &hkeyNetwork);
2550 if ((status != ERROR_SUCCESS) || !hkeyNetwork)
2551 SetErrBreak ((
2552 tr ("Host interface network is not found in registry (%s) [1]"),
2553 strRegLocation));
2554
2555 status = RegOpenKeyExA (hkeyNetwork, "Connection", 0,
2556 KEY_READ, &hkeyConnection);
2557 if ((status != ERROR_SUCCESS) || !hkeyConnection)
2558 SetErrBreak ((
2559 tr ("Host interface network is not found in registry (%s) [2]"),
2560 strRegLocation));
2561
2562 DWORD len = sizeof (lszPnPInstanceId);
2563 DWORD dwKeyType;
2564 status = RegQueryValueExW (hkeyConnection, L"PnPInstanceID", NULL,
2565 &dwKeyType, (LPBYTE) lszPnPInstanceId, &len);
2566 if ((status != ERROR_SUCCESS) || (dwKeyType != REG_SZ))
2567 SetErrBreak ((
2568 tr ("Host interface network is not found in registry (%s) [3]"),
2569 strRegLocation));
2570 }
2571 while (0);
2572
2573 if (hkeyConnection)
2574 RegCloseKey (hkeyConnection);
2575 if (hkeyNetwork)
2576 RegCloseKey (hkeyNetwork);
2577
2578 if (VBOX_FAILURE (vrc))
2579 break;
2580
2581 /*
2582 * Now we are going to enumerate all network devices and
2583 * wait until we encounter the right device instance ID
2584 */
2585
2586 HDEVINFO hDeviceInfo = INVALID_HANDLE_VALUE;
2587
2588 do
2589 {
2590 BOOL ok;
2591 DWORD ret = 0;
2592 GUID netGuid;
2593 SP_DEVINFO_DATA DeviceInfoData;
2594 DWORD index = 0;
2595 BOOL found = FALSE;
2596 DWORD size = 0;
2597
2598 /* initialize the structure size */
2599 DeviceInfoData.cbSize = sizeof (SP_DEVINFO_DATA);
2600
2601 /* copy the net class GUID */
2602 memcpy (&netGuid, &GUID_DEVCLASS_NET, sizeof (GUID_DEVCLASS_NET));
2603
2604 /* return a device info set contains all installed devices of the Net class */
2605 hDeviceInfo = SetupDiGetClassDevs (&netGuid, NULL, NULL, DIGCF_PRESENT);
2606
2607 if (hDeviceInfo == INVALID_HANDLE_VALUE)
2608 SetErrBreak (("SetupDiGetClassDevs failed (0x%08X)", GetLastError()));
2609
2610 /* enumerate the driver info list */
2611 while (TRUE)
2612 {
2613 TCHAR *deviceHwid;
2614
2615 ok = SetupDiEnumDeviceInfo (hDeviceInfo, index, &DeviceInfoData);
2616
2617 if (!ok)
2618 {
2619 if (GetLastError() == ERROR_NO_MORE_ITEMS)
2620 break;
2621 else
2622 {
2623 index++;
2624 continue;
2625 }
2626 }
2627
2628 /* try to get the hardware ID registry property */
2629 ok = SetupDiGetDeviceRegistryProperty (hDeviceInfo,
2630 &DeviceInfoData,
2631 SPDRP_HARDWAREID,
2632 NULL,
2633 NULL,
2634 0,
2635 &size);
2636 if (!ok)
2637 {
2638 if (GetLastError() != ERROR_INSUFFICIENT_BUFFER)
2639 {
2640 index++;
2641 continue;
2642 }
2643
2644 deviceHwid = (TCHAR *) malloc (size);
2645 ok = SetupDiGetDeviceRegistryProperty (hDeviceInfo,
2646 &DeviceInfoData,
2647 SPDRP_HARDWAREID,
2648 NULL,
2649 (PBYTE)deviceHwid,
2650 size,
2651 NULL);
2652 if (!ok)
2653 {
2654 free (deviceHwid);
2655 deviceHwid = NULL;
2656 index++;
2657 continue;
2658 }
2659 }
2660 else
2661 {
2662 /* something is wrong. This shouldn't have worked with a NULL buffer */
2663 index++;
2664 continue;
2665 }
2666
2667 for (TCHAR *t = deviceHwid;
2668 t && *t && t < &deviceHwid[size / sizeof(TCHAR)];
2669 t += _tcslen (t) + 1)
2670 {
2671 if (!_tcsicmp (DRIVERHWID, t))
2672 {
2673 /* get the device instance ID */
2674 TCHAR devID [MAX_DEVICE_ID_LEN];
2675 if (CM_Get_Device_ID(DeviceInfoData.DevInst,
2676 devID, MAX_DEVICE_ID_LEN, 0) == CR_SUCCESS)
2677 {
2678 /* compare to what we determined before */
2679 if (wcscmp(devID, lszPnPInstanceId) == 0)
2680 {
2681 found = TRUE;
2682 break;
2683 }
2684 }
2685 }
2686 }
2687
2688 if (deviceHwid)
2689 {
2690 free (deviceHwid);
2691 deviceHwid = NULL;
2692 }
2693
2694 if (found)
2695 break;
2696
2697 index++;
2698 }
2699
2700 if (found == FALSE)
2701 SetErrBreak ((tr ("Host Interface Network driver not found (0x%08X)"),
2702 GetLastError()));
2703
2704 ok = SetupDiSetSelectedDevice (hDeviceInfo, &DeviceInfoData);
2705 if (!ok)
2706 SetErrBreak (("SetupDiSetSelectedDevice failed (0x%08X)",
2707 GetLastError()));
2708
2709 ok = SetupDiCallClassInstaller (DIF_REMOVE, hDeviceInfo, &DeviceInfoData);
2710 if (!ok)
2711 SetErrBreak (("SetupDiCallClassInstaller (DIF_REMOVE) failed (0x%08X)",
2712 GetLastError()));
2713 }
2714 while (0);
2715
2716 /* clean up the device info set */
2717 if (hDeviceInfo != INVALID_HANDLE_VALUE)
2718 SetupDiDestroyDeviceInfoList (hDeviceInfo);
2719
2720 if (VBOX_FAILURE (vrc))
2721 break;
2722 }
2723 while (0);
2724
2725 LogFlowFunc (("vrc=%Vrc\n", vrc));
2726 LogFlowFuncLeave();
2727 return vrc;
2728}
2729
2730#undef SetErrBreak
2731
2732/* static */
2733HRESULT Host::networkInterfaceHelperClient (SVCHlpClient *aClient,
2734 Progress *aProgress,
2735 void *aUser, int *aVrc)
2736{
2737 LogFlowFuncEnter();
2738 LogFlowFunc (("aClient={%p}, aProgress={%p}, aUser={%p}\n",
2739 aClient, aProgress, aUser));
2740
2741 AssertReturn ((aClient == NULL && aProgress == NULL && aVrc == NULL) ||
2742 (aClient != NULL && aProgress != NULL && aVrc != NULL),
2743 E_POINTER);
2744 AssertReturn (aUser, E_POINTER);
2745
2746 std::auto_ptr <NetworkInterfaceHelperClientData>
2747 d (static_cast <NetworkInterfaceHelperClientData *> (aUser));
2748
2749 if (aClient == NULL)
2750 {
2751 /* "cleanup only" mode, just return (it will free aUser) */
2752 return S_OK;
2753 }
2754
2755 HRESULT rc = S_OK;
2756 int vrc = VINF_SUCCESS;
2757
2758 switch (d->msgCode)
2759 {
2760 case SVCHlpMsg::CreateHostNetworkInterface:
2761 {
2762 LogFlowFunc (("CreateHostNetworkInterface:\n"));
2763 LogFlowFunc (("Network connection name = '%ls'\n", d->name.raw()));
2764
2765 /* write message and parameters */
2766 vrc = aClient->write (d->msgCode);
2767 if (VBOX_FAILURE (vrc)) break;
2768 vrc = aClient->write (Utf8Str (d->name));
2769 if (VBOX_FAILURE (vrc)) break;
2770
2771 /* wait for a reply */
2772 bool endLoop = false;
2773 while (!endLoop)
2774 {
2775 SVCHlpMsg::Code reply = SVCHlpMsg::Null;
2776
2777 vrc = aClient->read (reply);
2778 if (VBOX_FAILURE (vrc)) break;
2779
2780 switch (reply)
2781 {
2782 case SVCHlpMsg::CreateHostNetworkInterface_OK:
2783 {
2784 /* read the GUID */
2785 Guid guid;
2786 vrc = aClient->read (guid);
2787 if (VBOX_FAILURE (vrc)) break;
2788
2789 LogFlowFunc (("Network connection GUID = {%Vuuid}\n", guid.raw()));
2790
2791 /* initialize the object returned to the caller by
2792 * CreateHostNetworkInterface() */
2793 rc = d->iface->init (d->name, guid);
2794 endLoop = true;
2795 break;
2796 }
2797 case SVCHlpMsg::Error:
2798 {
2799 /* read the error message */
2800 Utf8Str errMsg;
2801 vrc = aClient->read (errMsg);
2802 if (VBOX_FAILURE (vrc)) break;
2803
2804 rc = setError (E_FAIL, errMsg);
2805 endLoop = true;
2806 break;
2807 }
2808 default:
2809 {
2810 endLoop = true;
2811 ComAssertMsgFailedBreak ((
2812 "Invalid message code %d (%08lX)\n",
2813 reply, reply),
2814 rc = E_FAIL);
2815 }
2816 }
2817 }
2818
2819 break;
2820 }
2821 case SVCHlpMsg::RemoveHostNetworkInterface:
2822 {
2823 LogFlowFunc (("RemoveHostNetworkInterface:\n"));
2824 LogFlowFunc (("Network connection GUID = {%Vuuid}\n", d->guid.raw()));
2825
2826 /* write message and parameters */
2827 vrc = aClient->write (d->msgCode);
2828 if (VBOX_FAILURE (vrc)) break;
2829 vrc = aClient->write (d->guid);
2830 if (VBOX_FAILURE (vrc)) break;
2831
2832 /* wait for a reply */
2833 bool endLoop = false;
2834 while (!endLoop)
2835 {
2836 SVCHlpMsg::Code reply = SVCHlpMsg::Null;
2837
2838 vrc = aClient->read (reply);
2839 if (VBOX_FAILURE (vrc)) break;
2840
2841 switch (reply)
2842 {
2843 case SVCHlpMsg::OK:
2844 {
2845 /* no parameters */
2846 rc = S_OK;
2847 endLoop = true;
2848 break;
2849 }
2850 case SVCHlpMsg::Error:
2851 {
2852 /* read the error message */
2853 Utf8Str errMsg;
2854 vrc = aClient->read (errMsg);
2855 if (VBOX_FAILURE (vrc)) break;
2856
2857 rc = setError (E_FAIL, errMsg);
2858 endLoop = true;
2859 break;
2860 }
2861 default:
2862 {
2863 endLoop = true;
2864 ComAssertMsgFailedBreak ((
2865 "Invalid message code %d (%08lX)\n",
2866 reply, reply),
2867 rc = E_FAIL);
2868 }
2869 }
2870 }
2871
2872 break;
2873 }
2874 default:
2875 ComAssertMsgFailedBreak ((
2876 "Invalid message code %d (%08lX)\n",
2877 d->msgCode, d->msgCode),
2878 rc = E_FAIL);
2879 }
2880
2881 if (aVrc)
2882 *aVrc = vrc;
2883
2884 LogFlowFunc (("rc=0x%08X, vrc=%Vrc\n", rc, vrc));
2885 LogFlowFuncLeave();
2886 return rc;
2887}
2888
2889/* static */
2890int Host::networkInterfaceHelperServer (SVCHlpClient *aClient,
2891 SVCHlpMsg::Code aMsgCode)
2892{
2893 LogFlowFuncEnter();
2894 LogFlowFunc (("aClient={%p}, aMsgCode=%d\n", aClient, aMsgCode));
2895
2896 AssertReturn (aClient, VERR_INVALID_POINTER);
2897
2898 int vrc = VINF_SUCCESS;
2899
2900 switch (aMsgCode)
2901 {
2902 case SVCHlpMsg::CreateHostNetworkInterface:
2903 {
2904 LogFlowFunc (("CreateHostNetworkInterface:\n"));
2905
2906 Utf8Str name;
2907 vrc = aClient->read (name);
2908 if (VBOX_FAILURE (vrc)) break;
2909
2910 Guid guid;
2911 Utf8Str errMsg;
2912 vrc = createNetworkInterface (aClient, name, guid, errMsg);
2913
2914 if (VBOX_SUCCESS (vrc))
2915 {
2916 /* write success followed by GUID */
2917 vrc = aClient->write (SVCHlpMsg::CreateHostNetworkInterface_OK);
2918 if (VBOX_FAILURE (vrc)) break;
2919 vrc = aClient->write (guid);
2920 if (VBOX_FAILURE (vrc)) break;
2921 }
2922 else
2923 {
2924 /* write failure followed by error message */
2925 if (errMsg.isEmpty())
2926 errMsg = Utf8StrFmt ("Unspecified error (%Vrc)", vrc);
2927 vrc = aClient->write (SVCHlpMsg::Error);
2928 if (VBOX_FAILURE (vrc)) break;
2929 vrc = aClient->write (errMsg);
2930 if (VBOX_FAILURE (vrc)) break;
2931 }
2932
2933 break;
2934 }
2935 case SVCHlpMsg::RemoveHostNetworkInterface:
2936 {
2937 LogFlowFunc (("RemoveHostNetworkInterface:\n"));
2938
2939 Guid guid;
2940 vrc = aClient->read (guid);
2941 if (VBOX_FAILURE (vrc)) break;
2942
2943 Utf8Str errMsg;
2944 vrc = removeNetworkInterface (aClient, guid, errMsg);
2945
2946 if (VBOX_SUCCESS (vrc))
2947 {
2948 /* write parameter-less success */
2949 vrc = aClient->write (SVCHlpMsg::OK);
2950 if (VBOX_FAILURE (vrc)) break;
2951 }
2952 else
2953 {
2954 /* write failure followed by error message */
2955 if (errMsg.isEmpty())
2956 errMsg = Utf8StrFmt ("Unspecified error (%Vrc)", vrc);
2957 vrc = aClient->write (SVCHlpMsg::Error);
2958 if (VBOX_FAILURE (vrc)) break;
2959 vrc = aClient->write (errMsg);
2960 if (VBOX_FAILURE (vrc)) break;
2961 }
2962
2963 break;
2964 }
2965 default:
2966 AssertMsgFailedBreakStmt (
2967 ("Invalid message code %d (%08lX)\n", aMsgCode, aMsgCode),
2968 VERR_GENERAL_FAILURE);
2969 }
2970
2971 LogFlowFunc (("vrc=%Vrc\n", vrc));
2972 LogFlowFuncLeave();
2973 return vrc;
2974}
2975
2976#endif /* RT_OS_WINDOWS */
2977
2978#ifdef VBOX_WITH_RESOURCE_USAGE_API
2979void Host::registerMetrics (PerformanceCollector *aCollector)
2980{
2981 pm::CollectorHAL *hal = aCollector->getHAL();
2982 /* Create sub metrics */
2983 pm::SubMetric *cpuLoadUser = new pm::SubMetric ("CPU/Load/User",
2984 "Percentage of processor time spent in user mode.");
2985 pm::SubMetric *cpuLoadKernel = new pm::SubMetric ("CPU/Load/Kernel",
2986 "Percentage of processor time spent in kernel mode.");
2987 pm::SubMetric *cpuLoadIdle = new pm::SubMetric ("CPU/Load/Idle",
2988 "Percentage of processor time spent idling.");
2989 pm::SubMetric *cpuMhzSM = new pm::SubMetric ("CPU/MHz",
2990 "Average of current frequency of all processors.");
2991 pm::SubMetric *ramUsageTotal = new pm::SubMetric ("RAM/Usage/Total",
2992 "Total physical memory installed.");
2993 pm::SubMetric *ramUsageUsed = new pm::SubMetric ("RAM/Usage/Used",
2994 "Physical memory currently occupied.");
2995 pm::SubMetric *ramUsageFree = new pm::SubMetric ("RAM/Usage/Free",
2996 "Physical memory currently available to applications.");
2997 /* Create and register base metrics */
2998 IUnknown *objptr;
2999 ComObjPtr <Host> tmp = this;
3000 tmp.queryInterfaceTo (&objptr);
3001 pm::BaseMetric *cpuLoad = new pm::HostCpuLoadRaw (hal, objptr, cpuLoadUser, cpuLoadKernel,
3002 cpuLoadIdle);
3003 aCollector->registerBaseMetric (cpuLoad);
3004 pm::BaseMetric *cpuMhz = new pm::HostCpuMhz (hal, objptr, cpuMhzSM);
3005 aCollector->registerBaseMetric (cpuMhz);
3006 pm::BaseMetric *ramUsage = new pm::HostRamUsage (hal, objptr, ramUsageTotal, ramUsageUsed,
3007 ramUsageFree);
3008 aCollector->registerBaseMetric (ramUsage);
3009
3010 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadUser, 0));
3011 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadUser,
3012 new pm::AggregateAvg()));
3013 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadUser,
3014 new pm::AggregateMin()));
3015 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadUser,
3016 new pm::AggregateMax()));
3017
3018 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadKernel, 0));
3019 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadKernel,
3020 new pm::AggregateAvg()));
3021 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadKernel,
3022 new pm::AggregateMin()));
3023 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadKernel,
3024 new pm::AggregateMax()));
3025
3026 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadIdle, 0));
3027 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadIdle,
3028 new pm::AggregateAvg()));
3029 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadIdle,
3030 new pm::AggregateMin()));
3031 aCollector->registerMetric (new pm::Metric(cpuLoad, cpuLoadIdle,
3032 new pm::AggregateMax()));
3033
3034 aCollector->registerMetric (new pm::Metric(cpuMhz, cpuMhzSM, 0));
3035 aCollector->registerMetric (new pm::Metric(cpuMhz, cpuMhzSM,
3036 new pm::AggregateAvg()));
3037 aCollector->registerMetric (new pm::Metric(cpuMhz, cpuMhzSM,
3038 new pm::AggregateMin()));
3039 aCollector->registerMetric (new pm::Metric(cpuMhz, cpuMhzSM,
3040 new pm::AggregateMax()));
3041
3042 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageTotal, 0));
3043 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageTotal,
3044 new pm::AggregateAvg()));
3045 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageTotal,
3046 new pm::AggregateMin()));
3047 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageTotal,
3048 new pm::AggregateMax()));
3049
3050 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageUsed, 0));
3051 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageUsed,
3052 new pm::AggregateAvg()));
3053 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageUsed,
3054 new pm::AggregateMin()));
3055 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageUsed,
3056 new pm::AggregateMax()));
3057
3058 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageFree, 0));
3059 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageFree,
3060 new pm::AggregateAvg()));
3061 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageFree,
3062 new pm::AggregateMin()));
3063 aCollector->registerMetric (new pm::Metric(ramUsage, ramUsageFree,
3064 new pm::AggregateMax()));
3065};
3066
3067void Host::unregisterMetrics (PerformanceCollector *aCollector)
3068{
3069 aCollector->unregisterMetricsFor (this);
3070 aCollector->unregisterBaseMetricsFor (this);
3071};
3072#endif /* VBOX_WITH_RESOURCE_USAGE_API */
3073
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