VirtualBox

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

Last change on this file since 10146 was 9937, checked in by vboxsync, 17 years ago

Main: Corrected getProcessorUsage() impl (docs, spacing etc).

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