VirtualBox

source: vbox/trunk/src/VBox/Main/linux/HostHardwareLinux.cpp@ 26270

Last change on this file since 26270 was 26186, checked in by vboxsync, 15 years ago

Main: coding style fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 72.9 KB
Line 
1/* $Id: HostHardwareLinux.cpp 26186 2010-02-03 13:07:12Z vboxsync $ */
2/** @file
3 * Classes for handling hardware detection under Linux. Please feel free to
4 * expand these to work for other systems (Solaris!) or to add new ones for
5 * other systems.
6 */
7
8/*
9 * Copyright (C) 2008 Sun Microsystems, Inc.
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 *
19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
20 * Clara, CA 95054 USA or visit http://www.sun.com if you need
21 * additional information or have any questions.
22 */
23
24#define LOG_GROUP LOG_GROUP_MAIN
25
26/*******************************************************************************
27* Header Files *
28*******************************************************************************/
29
30#include <HostHardwareLinux.h>
31
32#include <VBox/log.h>
33# ifdef VBOX_WITH_DBUS
34# include <VBox/dbus.h>
35# endif
36
37#include <iprt/dir.h>
38#include <iprt/env.h>
39#include <iprt/file.h>
40#include <iprt/mem.h>
41#include <iprt/param.h>
42#include <iprt/path.h>
43#include <iprt/thread.h> /* for RTThreadSleep() */
44#include <iprt/string.h>
45
46#ifdef RT_OS_LINUX
47# include <sys/types.h>
48# include <sys/stat.h>
49# include <unistd.h>
50# include <fcntl.h>
51/* bird: This is a hack to work around conflicts between these linux kernel headers
52 * and the GLIBC tcpip headers. They have different declarations of the 4
53 * standard byte order functions. */
54// # define _LINUX_BYTEORDER_GENERIC_H
55# define _LINUX_BYTEORDER_SWABB_H
56# include <linux/cdrom.h>
57# include <linux/fd.h>
58# include <linux/major.h>
59# include <errno.h>
60# include <scsi/scsi.h>
61
62# include <iprt/linux/sysfs.h>
63#endif /* RT_OS_LINUX */
64#include <vector>
65
66/******************************************************************************
67* Global Variables *
68******************************************************************************/
69
70#ifdef TESTCASE
71static bool testing() { return true; }
72static bool fNoProbe = false;
73static bool noProbe() { return fNoProbe; }
74static void setNoProbe(bool val) { fNoProbe = val; }
75#else
76static bool testing() { return false; }
77static bool noProbe() { return false; }
78static void setNoProbe(bool val) { (void)val; }
79#endif
80
81/******************************************************************************
82* Typedefs and Defines *
83******************************************************************************/
84
85/** When waiting for hotplug events, we currently restart the wait after at
86 * most this many milliseconds. */
87enum { DBUS_POLL_TIMEOUT = 2000 /* ms */ };
88
89static int getDriveInfoFromEnv(const char *pcszVar, DriveInfoList *pList,
90 bool isDVD, bool *pfSuccess);
91static int getDriveInfoFromDev(DriveInfoList *pList, bool isDVD,
92 bool *pfSuccess);
93static int getDriveInfoFromSysfs(DriveInfoList *pList, bool isDVD,
94 bool *pfSuccess);
95#ifdef VBOX_WITH_DBUS
96/* These must be extern to be usable in the RTMemAutoPtr template */
97extern void VBoxHalShutdown (DBusConnection *pConnection);
98extern void VBoxHalShutdownPrivate (DBusConnection *pConnection);
99extern void VBoxDBusConnectionUnref(DBusConnection *pConnection);
100extern void VBoxDBusConnectionCloseAndUnref(DBusConnection *pConnection);
101extern void VBoxDBusMessageUnref(DBusMessage *pMessage);
102
103static int halInit(RTMemAutoPtr <DBusConnection, VBoxHalShutdown> *pConnection);
104static int halInitPrivate(RTMemAutoPtr <DBusConnection, VBoxHalShutdownPrivate> *pConnection);
105static int halFindDeviceStringMatch (DBusConnection *pConnection,
106 const char *pszKey, const char *pszValue,
107 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> *pMessage);
108/*
109static int halFindDeviceStringMatchVector (DBusConnection *pConnection,
110 const char *pszKey,
111 const char *pszValue,
112 std::vector<iprt::MiniString> *pMatches);
113*/
114static int halGetPropertyStrings (DBusConnection *pConnection,
115 const char *pszUdi, size_t cKeys,
116 const char **papszKeys, char **papszValues,
117 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> *pMessage);
118/*
119static int halGetPropertyStringsVector (DBusConnection *pConnection,
120 const char *pszUdi, size_t cProps,
121 const char **papszKeys,
122 std::vector<iprt::MiniString> *pMatches,
123 bool *pfMatches, bool *pfSuccess);
124*/
125static int getUSBDeviceInfoFromHal(USBDeviceInfoList *pList, bool *pfSuccess);
126static int getOldUSBDeviceInfoFromHal(USBDeviceInfoList *pList, bool *pfSuccess);
127static int getUSBInterfacesFromHal(std::vector <iprt::MiniString> *pList,
128 const char *pcszUdi, bool *pfSuccess);
129static DBusHandlerResult dbusFilterFunction (DBusConnection *pConnection,
130 DBusMessage *pMessage, void *pvUser);
131#endif /* VBOX_WITH_DBUS */
132
133
134/** Find the length of a string, ignoring trailing non-ascii or control
135 * characters */
136static size_t strLenStripped(const char *pcsz)
137{
138 size_t cch = 0;
139 for (size_t i = 0; pcsz[i] != '\0'; ++i)
140 if (pcsz[i] > 32 && pcsz[i] < 127)
141 cch = i;
142 return cch + 1;
143}
144
145
146/**
147 * Get the name of a floppy drive according to the Linux floppy driver.
148 * @returns true on success, false if the name was not available (i.e. the
149 * device was not readible, or the file name wasn't a PC floppy
150 * device)
151 * @param pcszNode the path to the device node for the device
152 * @param Number the Linux floppy driver number for the drive. Required.
153 * @param pszName where to store the name retreived
154 */
155static bool floppyGetName(const char *pcszNode, unsigned Number,
156 floppy_drive_name pszName)
157{
158 AssertPtrReturn(pcszNode, false);
159 AssertPtrReturn(pszName, false);
160 AssertReturn(Number <= 7, false);
161 RTFILE File;
162 int rc = RTFileOpen(&File, pcszNode, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE | RTFILE_O_NON_BLOCK);
163 if (RT_SUCCESS(rc))
164 {
165 int rcIoCtl;
166 /** @todo The next line can produce a warning, as the ioctl request
167 * field is defined as signed, but the Linux ioctl definition macros
168 * produce unsigned constants. */
169 rc = RTFileIoCtl(File, FDGETDRVTYP, pszName, 0, &rcIoCtl);
170 RTFileClose(File);
171 if (RT_SUCCESS(rc) && rcIoCtl >= 0)
172 return true;
173 }
174 return false;
175}
176
177
178/**
179 * Create a UDI and a description for a floppy drive based on a number and the
180 * driver's name for it. We deliberately return an ugly sequence of
181 * characters as the description rather than an English language string to
182 * avoid translation issues.
183 *
184 * @returns true if we know the device to be valid, false otherwise
185 * @param pcszName the floppy driver name for the device (optional)
186 * @param Number the number of the floppy (0 to 3 on FDC 0, 4 to 7 on
187 * FDC 1)
188 * @param pszDesc where to store the device description (optional)
189 * @param cchDesc the size of the buffer in @a pszDesc
190 * @param pszUdi where to store the device UDI (optional)
191 * @param cchUdi the size of the buffer in @a pszUdi
192 */
193static void floppyCreateDeviceStrings(const floppy_drive_name pcszName,
194 unsigned Number, char *pszDesc,
195 size_t cchDesc, char *pszUdi,
196 size_t cchUdi)
197{
198 AssertPtrNullReturnVoid(pcszName);
199 AssertPtrNullReturnVoid(pszDesc);
200 AssertReturnVoid(!pszDesc || cchDesc > 0);
201 AssertPtrNullReturnVoid(pszUdi);
202 AssertReturnVoid(!pszUdi || cchUdi > 0);
203 AssertReturnVoid(Number <= 7);
204 if (pcszName)
205 {
206 const char *pcszSize;
207 switch(pcszName[0])
208 {
209 case 'd': case 'q': case 'h':
210 pcszSize = "5.25\"";
211 break;
212 case 'D': case 'H': case 'E': case 'u':
213 pcszSize = "3.5\"";
214 break;
215 default:
216 pcszSize = "(unknown)";
217 }
218 if (pszDesc)
219 RTStrPrintf(pszDesc, cchDesc, "%s %s K%s", pcszSize, &pcszName[1],
220 Number > 3 ? ", FDC 2" : "");
221 }
222 else
223 {
224 if (pszDesc)
225 RTStrPrintf(pszDesc, cchDesc, "FDD %d%s", (Number & 4) + 1,
226 Number > 3 ? ", FDC 2" : "");
227 }
228 if (pszUdi)
229 RTStrPrintf(pszUdi, cchUdi,
230 "/org/freedesktop/Hal/devices/platform_floppy_%u_storage",
231 Number);
232}
233
234
235/**
236 * Check whether a device number might correspond to a CD-ROM device according
237 * to Documentation/devices.txt in the Linux kernel source.
238 * @returns true if it might, false otherwise
239 * @param Number the device number (major and minor combination)
240 */
241static bool isCdromDevNum(dev_t Number)
242{
243 int major = major(Number);
244 int minor = minor(Number);
245 if ((major == IDE0_MAJOR) && !(minor & 0x3f))
246 return true;
247 if (major == SCSI_CDROM_MAJOR)
248 return true;
249 if (major == CDU31A_CDROM_MAJOR)
250 return true;
251 if (major == GOLDSTAR_CDROM_MAJOR)
252 return true;
253 if (major == OPTICS_CDROM_MAJOR)
254 return true;
255 if (major == SANYO_CDROM_MAJOR)
256 return true;
257 if (major == MITSUMI_X_CDROM_MAJOR)
258 return true;
259 if ((major == IDE1_MAJOR) && !(minor & 0x3f))
260 return true;
261 if (major == MITSUMI_CDROM_MAJOR)
262 return true;
263 if (major == CDU535_CDROM_MAJOR)
264 return true;
265 if (major == MATSUSHITA_CDROM_MAJOR)
266 return true;
267 if (major == MATSUSHITA_CDROM2_MAJOR)
268 return true;
269 if (major == MATSUSHITA_CDROM3_MAJOR)
270 return true;
271 if (major == MATSUSHITA_CDROM4_MAJOR)
272 return true;
273 if (major == AZTECH_CDROM_MAJOR)
274 return true;
275 if (major == 30 /* CM205_CDROM_MAJOR */) /* no #define for some reason */
276 return true;
277 if (major == CM206_CDROM_MAJOR)
278 return true;
279 if ((major == IDE3_MAJOR) && !(minor & 0x3f))
280 return true;
281 if (major == 46 /* Parallel port ATAPI CD-ROM */) /* no #define */
282 return true;
283 if ((major == IDE4_MAJOR) && !(minor & 0x3f))
284 return true;
285 if ((major == IDE5_MAJOR) && !(minor & 0x3f))
286 return true;
287 if ((major == IDE6_MAJOR) && !(minor & 0x3f))
288 return true;
289 if ((major == IDE7_MAJOR) && !(minor & 0x3f))
290 return true;
291 if ((major == IDE8_MAJOR) && !(minor & 0x3f))
292 return true;
293 if ((major == IDE9_MAJOR) && !(minor & 0x3f))
294 return true;
295 if (major == 113 /* VIOCD_MAJOR */)
296 return true;
297 return false;
298}
299
300
301/**
302 * Send an SCSI INQUIRY command to a device and return selected information.
303 * @returns iprt status code
304 * @returns VERR_TRY_AGAIN if the query failed but might succeed next time
305 * @param pcszNode the full path to the device node
306 * @param pu8Type where to store the SCSI device type on success (optional)
307 * @param pchVendor where to store the vendor id string on success (optional)
308 * @param cchVendor the size of the @a pchVendor buffer
309 * @param pchModel where to store the product id string on success (optional)
310 * @param cchModel the size of the @a pchModel buffer
311 * @note check documentation on the SCSI INQUIRY command and the Linux kernel
312 * SCSI headers included above if you want to understand what is going
313 * on in this method.
314 */
315static int cdromDoInquiry(const char *pcszNode, uint8_t *pu8Type,
316 char *pchVendor, size_t cchVendor, char *pchModel,
317 size_t cchModel)
318{
319 LogRelFlowFunc(("pcszNode=%s, pu8Type=%p, pchVendor=%p, cchVendor=%llu, pchModel=%p, cchModel=%llu\n",
320 pcszNode, pu8Type, pchVendor, cchVendor, pchModel,
321 cchModel));
322 AssertPtrReturn(pcszNode, VERR_INVALID_POINTER);
323 AssertPtrNullReturn(pu8Type, VERR_INVALID_POINTER);
324 AssertPtrNullReturn(pchVendor, VERR_INVALID_POINTER);
325 AssertPtrNullReturn(pchModel, VERR_INVALID_POINTER);
326
327 unsigned char u8Response[96] = { 0 };
328 struct cdrom_generic_command CdromCommandReq =
329 { { INQUIRY, 0, 0, 0, sizeof(u8Response), 0 } /* INQUIRY */ };
330 int rc, rcIoCtl = 0;
331 RTFILE file;
332 rc = RTFileOpen(&file, pcszNode, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE | RTFILE_O_NON_BLOCK);
333 if (RT_SUCCESS(rc))
334 {
335 CdromCommandReq.buffer = u8Response;
336 CdromCommandReq.buflen = sizeof(u8Response);
337 CdromCommandReq.data_direction = CGC_DATA_READ;
338 CdromCommandReq.timeout = 5000; /* ms */
339 rc = RTFileIoCtl(file, CDROM_SEND_PACKET, &CdromCommandReq, 0,
340 &rcIoCtl);
341 if (RT_SUCCESS(rc) && rcIoCtl < 0)
342 rc = RTErrConvertFromErrno(-CdromCommandReq.stat);
343 RTFileClose(file);
344 }
345 if (RT_SUCCESS(rc))
346 {
347 if (pu8Type)
348 *pu8Type = u8Response[0] & 0x1f;
349 if (pchVendor)
350 RTStrPrintf(pchVendor, cchVendor, "%.8s",
351 (char *) &u8Response[8] /* vendor id string */);
352 if (pchModel)
353 RTStrPrintf(pchModel, cchModel, "%.16s",
354 (char *) &u8Response[16] /* product id string */);
355 }
356 LogRelFlowFunc(("returning %Rrc\n", rc));
357 if (RT_SUCCESS(rc))
358 LogRelFlowFunc((" type=%u, vendor=%.8s, product=%.16s\n",
359 u8Response[0] & 0x1f, (char *) &u8Response[8],
360 (char *) &u8Response[16]));
361 return rc;
362}
363
364
365/**
366 * Initialise the device strings (description and UDI) for a DVD drive based on
367 * vendor and model name strings.
368 * @param pcszVendor the vendor ID string
369 * @param pcszModel the product ID string
370 * @param pszDesc where to store the description string (optional)
371 * @param cchDesc the size of the buffer in @pszDesc
372 * @param pszUdi where to store the UDI string (optional)
373 * @param cchUdi the size of the buffer in @pszUdi
374 */
375/* static */
376void dvdCreateDeviceStrings(const char *pcszVendor, const char *pcszModel,
377 char *pszDesc, size_t cchDesc, char *pszUdi,
378 size_t cchUdi)
379{
380 AssertPtrReturnVoid(pcszVendor);
381 AssertPtrReturnVoid(pcszModel);
382 AssertPtrNullReturnVoid(pszDesc);
383 AssertReturnVoid(!pszDesc || cchDesc > 0);
384 AssertPtrNullReturnVoid(pszUdi);
385 AssertReturnVoid(!pszUdi || cchUdi > 0);
386 char szCleaned[128];
387 size_t cchVendor = strLenStripped(pcszVendor);
388 size_t cchModel = strLenStripped(pcszModel);
389
390 /* Create a cleaned version of the model string for the UDI string. */
391 for (unsigned i = 0; pcszModel[i] != '\0' && i < sizeof(szCleaned); ++i)
392 if ( (pcszModel[i] >= '0' && pcszModel[i] <= '9')
393 || (pcszModel[i] >= 'A' && pcszModel[i] <= 'z'))
394 szCleaned[i] = pcszModel[i];
395 else
396 szCleaned[i] = '_';
397 szCleaned[RT_MIN(cchModel, sizeof(szCleaned) - 1)] = '\0';
398
399 /* Construct the description string as "Vendor Product" */
400 if (pszDesc)
401 {
402 if (cchVendor > 0)
403 RTStrPrintf(pszDesc, cchDesc, "%.*s %s", cchVendor, pcszVendor,
404 cchModel > 0 ? pcszModel : "(unknown drive model)");
405 else
406 RTStrPrintf(pszDesc, cchDesc, "%s", pcszModel);
407 }
408 /* Construct the UDI string */
409 if (pszUdi)
410 {
411 if (cchModel > 0)
412 RTStrPrintf(pszUdi, cchUdi,
413 "/org/freedesktop/Hal/devices/storage_model_%s",
414 szCleaned);
415 else
416 pszUdi[0] = '\0';
417 }
418}
419
420
421/**
422 * Check whether a device node points to a valid device and create a UDI and
423 * a description for it, and store the device number, if it does.
424 * @returns true if the device is valid, false otherwise
425 * @param pcszNode the path to the device node
426 * @param isDVD are we looking for a DVD device (or a floppy device)?
427 * @param pDevice where to store the device node (optional)
428 * @param pszDesc where to store the device description (optional)
429 * @param cchDesc the size of the buffer in @a pszDesc
430 * @param pszUdi where to store the device UDI (optional)
431 * @param cchUdi the size of the buffer in @a pszUdi
432 */
433static bool devValidateDevice(const char *pcszNode, bool isDVD, dev_t *pDevice,
434 char *pszDesc, size_t cchDesc, char *pszUdi,
435 size_t cchUdi)
436{
437 AssertPtrReturn(pcszNode, false);
438 AssertPtrNullReturn(pDevice, false);
439 AssertPtrNullReturn(pszDesc, false);
440 AssertReturn(!pszDesc || cchDesc > 0, false);
441 AssertPtrNullReturn(pszUdi, false);
442 AssertReturn(!pszUdi || cchUdi > 0, false);
443 RTFSOBJINFO ObjInfo;
444 if (RT_FAILURE(RTPathQueryInfo(pcszNode, &ObjInfo, RTFSOBJATTRADD_UNIX)))
445 return false;
446 if (!RTFS_IS_DEV_BLOCK(ObjInfo.Attr.fMode))
447 return false;
448 if (pDevice)
449 *pDevice = ObjInfo.Attr.u.Unix.Device;
450 if (isDVD)
451 {
452 char szVendor[128], szModel[128];
453 uint8_t u8Type;
454 if (!isCdromDevNum(ObjInfo.Attr.u.Unix.Device))
455 return false;
456 if (RT_FAILURE(cdromDoInquiry(pcszNode, &u8Type,
457 szVendor, sizeof(szVendor),
458 szModel, sizeof(szModel))))
459 return false;
460 if (u8Type != TYPE_ROM)
461 return false;
462 dvdCreateDeviceStrings(szVendor, szModel, pszDesc, cchDesc,
463 pszUdi, cchUdi);
464 }
465 else
466 {
467 /* Floppies on Linux are legacy devices with hardcoded majors and
468 * minors */
469 unsigned Number;
470 floppy_drive_name szName;
471 if (major(ObjInfo.Attr.u.Unix.Device) != FLOPPY_MAJOR)
472 return false;
473 switch (minor(ObjInfo.Attr.u.Unix.Device))
474 {
475 case 0: case 1: case 2: case 3:
476 Number = minor(ObjInfo.Attr.u.Unix.Device);
477 break;
478 case 128: case 129: case 130: case 131:
479 Number = minor(ObjInfo.Attr.u.Unix.Device) - 128 + 4;
480 break;
481 default:
482 return false;
483 }
484 if (!floppyGetName(pcszNode, Number, szName))
485 return false;
486 floppyCreateDeviceStrings(szName, Number, pszDesc, cchDesc, pszUdi,
487 cchUdi);
488 }
489 return true;
490}
491
492
493int VBoxMainDriveInfo::updateDVDs ()
494{
495 LogFlowThisFunc(("entered\n"));
496 int rc = VINF_SUCCESS;
497 bool success = false; /* Have we succeeded in finding anything yet? */
498 try
499 {
500 mDVDList.clear ();
501 /* Always allow the user to override our auto-detection using an
502 * environment variable. */
503 if (RT_SUCCESS(rc) && (!success || testing()))
504 rc = getDriveInfoFromEnv ("VBOX_CDROM", &mDVDList, true /* isDVD */,
505 &success);
506 setNoProbe(false);
507 if (RT_SUCCESS(rc) && (!success | testing()))
508 rc = getDriveInfoFromSysfs(&mDVDList, true /* isDVD */, &success);
509 if (RT_SUCCESS(rc) && testing())
510 {
511 setNoProbe(true);
512 rc = getDriveInfoFromSysfs(&mDVDList, true /* isDVD */, &success);
513 }
514 /* Walk through the /dev subtree if nothing else has helped. */
515 if (RT_SUCCESS(rc) && (!success | testing()))
516 rc = getDriveInfoFromDev(&mDVDList, true /* isDVD */, &success);
517 }
518 catch(std::bad_alloc &e)
519 {
520 rc = VERR_NO_MEMORY;
521 }
522 LogFlowThisFunc(("rc=%Rrc\n", rc));
523 return rc;
524}
525
526int VBoxMainDriveInfo::updateFloppies ()
527{
528 LogFlowThisFunc(("entered\n"));
529 int rc = VINF_SUCCESS;
530 bool success = false; /* Have we succeeded in finding anything yet? */
531 try
532 {
533 mFloppyList.clear ();
534 if (RT_SUCCESS(rc) && (!success || testing()))
535 rc = getDriveInfoFromEnv("VBOX_FLOPPY", &mFloppyList,
536 false /* isDVD */, &success);
537 setNoProbe(false);
538 if ( RT_SUCCESS(rc) && (!success || testing()))
539 rc = getDriveInfoFromSysfs(&mFloppyList, false /* isDVD */,
540 &success);
541 if (RT_SUCCESS(rc) && testing())
542 {
543 setNoProbe(true);
544 rc = getDriveInfoFromSysfs(&mFloppyList, false /* isDVD */, &success);
545 }
546 /* Walk through the /dev subtree if nothing else has helped. */
547 if ( RT_SUCCESS(rc) && (!success || testing()))
548 rc = getDriveInfoFromDev(&mFloppyList, false /* isDVD */,
549 &success);
550 }
551 catch(std::bad_alloc &e)
552 {
553 rc = VERR_NO_MEMORY;
554 }
555 LogFlowThisFunc(("rc=%Rrc\n", rc));
556 return rc;
557}
558
559
560/**
561 * Extract the names of drives from an environment variable and add them to a
562 * list if they are valid.
563 * @returns iprt status code
564 * @param pcszVar the name of the environment variable. The variable
565 * value should be a list of device node names, separated
566 * by ':' characters.
567 * @param pList the list to append the drives found to
568 * @param isDVD are we looking for DVD drives or for floppies?
569 * @param pfSuccess this will be set to true if we found at least one drive
570 * and to false otherwise. Optional.
571 */
572/* static */
573int getDriveInfoFromEnv(const char *pcszVar, DriveInfoList *pList,
574 bool isDVD, bool *pfSuccess)
575{
576 AssertPtrReturn(pcszVar, VERR_INVALID_POINTER);
577 AssertPtrReturn(pList, VERR_INVALID_POINTER);
578 AssertPtrNullReturn(pfSuccess, VERR_INVALID_POINTER);
579 LogFlowFunc(("pcszVar=%s, pList=%p, isDVD=%d, pfSuccess=%p\n", pcszVar,
580 pList, isDVD, pfSuccess));
581 int rc = VINF_SUCCESS;
582 bool success = false;
583 char *pszFreeMe = RTEnvDupEx(RTENV_DEFAULT, pcszVar);
584
585 try
586 {
587 const char *pcszCurrent = pszFreeMe;
588 while (pcszCurrent && *pcszCurrent != '\0')
589 {
590 const char *pcszNext = strchr(pcszCurrent, ':');
591 char szPath[RTPATH_MAX], szReal[RTPATH_MAX];
592 char szDesc[256], szUdi[256];
593 if (pcszNext)
594 RTStrPrintf(szPath, sizeof(szPath), "%.*s",
595 pcszNext - pcszCurrent - 1, pcszCurrent);
596 else
597 RTStrPrintf(szPath, sizeof(szPath), "%s", pcszCurrent);
598 if ( RT_SUCCESS(RTPathReal(szPath, szReal, sizeof(szReal)))
599 && devValidateDevice(szReal, isDVD, NULL, szDesc,
600 sizeof(szDesc), szUdi, sizeof(szUdi)))
601 {
602 pList->push_back(DriveInfo(szReal, szUdi, szDesc));
603 success = true;
604 }
605 pcszCurrent = pcszNext ? pcszNext + 1 : NULL;
606 }
607 if (pfSuccess != NULL)
608 *pfSuccess = success;
609 }
610 catch(std::bad_alloc &e)
611 {
612 rc = VERR_NO_MEMORY;
613 }
614 RTStrFree(pszFreeMe);
615 LogFlowFunc(("rc=%Rrc, success=%d\n", rc, success));
616 return rc;
617}
618
619
620class sysfsBlockDev
621{
622public:
623 sysfsBlockDev(const char *pcszName, bool wantDVD)
624 : mpcszName(pcszName), mwantDVD(wantDVD), misConsistent(true),
625 misValid(false)
626 {
627 if (findDeviceNode())
628 {
629 if (mwantDVD)
630 validateAndInitForDVD();
631 else
632 validateAndInitForFloppy();
633 }
634 }
635private:
636 /** The name of the subdirectory of /sys/block for this device */
637 const char *mpcszName;
638 /** Are we looking for a floppy or a DVD device? */
639 bool mwantDVD;
640 /** The device node for the device */
641 char mszNode[RTPATH_MAX];
642 /** Does the sysfs entry look like we expect it too? This is a canary
643 * for future sysfs ABI changes. */
644 bool misConsistent;
645 /** Is this entry a valid specimen of what we are looking for? */
646 bool misValid;
647 /** Human readible drive description string */
648 char mszDesc[256];
649 /** Unique identifier for the drive. Should be identical to hal's UDI for
650 * the device. May not be unique for two identical drives. */
651 char mszUdi[256];
652private:
653 /* Private methods */
654
655 /**
656 * Fill in the device node member based on the /sys/block subdirectory.
657 * @returns boolean success value
658 */
659 bool findDeviceNode()
660 {
661 dev_t dev = RTLinuxSysFsReadDevNumFile("block/%s/dev", mpcszName);
662 if (dev == 0)
663 {
664 misConsistent = false;
665 return false;
666 }
667 if (RTLinuxFindDevicePath(dev, RTFS_TYPE_DEV_BLOCK, mszNode,
668 sizeof(mszNode), "%s", mpcszName) < 0)
669 return false;
670 return true;
671 }
672
673 /** Check whether the sysfs block entry is valid for a DVD device and
674 * initialise the string data members for the object. We try to get all
675 * the information we need from sysfs if possible, to avoid unnecessarily
676 * poking the device, and if that fails we fall back to an SCSI INQUIRY
677 * command. */
678 void validateAndInitForDVD()
679 {
680 char szVendor[128], szModel[128];
681 ssize_t cchVendor, cchModel;
682 int64_t type = RTLinuxSysFsReadIntFile(10, "block/%s/device/type",
683 mpcszName);
684 if (type >= 0 && type != TYPE_ROM)
685 return;
686 if (type == TYPE_ROM)
687 {
688 cchVendor = RTLinuxSysFsReadStrFile(szVendor, sizeof(szVendor),
689 "block/%s/device/vendor",
690 mpcszName);
691 if (cchVendor >= 0)
692 {
693 cchModel = RTLinuxSysFsReadStrFile(szModel, sizeof(szModel),
694 "block/%s/device/model",
695 mpcszName);
696 if (cchModel >= 0)
697 {
698 misValid = true;
699 dvdCreateDeviceStrings(szVendor, szModel,
700 mszDesc, sizeof(mszDesc),
701 mszUdi, sizeof(mszUdi));
702 return;
703 }
704 }
705 }
706 if (!noProbe())
707 probeAndInitForDVD();
708 }
709
710 /** Try to find out whether a device is a DVD drive by sending it an
711 * SCSI INQUIRY command. If it is, initialise the string and validity
712 * data members for the object based on the returned data.
713 */
714 void probeAndInitForDVD()
715 {
716 AssertReturnVoid(mszNode[0] != '\0');
717 uint8_t u8Type = 0;
718 char szVendor[128] = "";
719 char szModel[128] = "";
720 int rc = cdromDoInquiry(mszNode, &u8Type, szVendor,
721 sizeof(szVendor), szModel,
722 sizeof(szModel));
723 if (RT_SUCCESS(rc) && (u8Type == TYPE_ROM))
724 {
725 misValid = true;
726 dvdCreateDeviceStrings(szVendor, szModel, mszDesc, sizeof(mszDesc),
727 mszUdi, sizeof(mszUdi));
728 }
729 }
730
731 /** Check whether the sysfs block entry is valid for a floppy device and
732 * initialise the string data members for the object. Since we only
733 * support floppies using the basic "floppy" driver, we check the driver
734 * using the entry name and a driver-specific ioctl. */
735 void validateAndInitForFloppy()
736 {
737 bool haveName = false;
738 floppy_drive_name szName;
739 char szDriver[8];
740 if ( mpcszName[0] != 'f'
741 || mpcszName[1] != 'd'
742 || mpcszName[2] < '0'
743 || mpcszName[2] > '7'
744 || mpcszName[3] != '\0')
745 return;
746 if (!noProbe())
747 haveName = floppyGetName(mszNode, mpcszName[2] - '0', szName);
748 if (RTLinuxSysFsGetLinkDest(szDriver, sizeof(szDriver), "block/%s/%s",
749 mpcszName, "device/driver") >= 0)
750 {
751 if (RTStrCmp(szDriver, "floppy"))
752 return;
753 }
754 else if (!haveName)
755 return;
756 floppyCreateDeviceStrings(haveName ? szName : NULL,
757 mpcszName[2] - '0', mszDesc,
758 sizeof(mszDesc), mszUdi, sizeof(mszUdi));
759 misValid = true;
760 }
761
762public:
763 bool isConsistent()
764 {
765 return misConsistent;
766 }
767 bool isValid()
768 {
769 return misValid;
770 }
771 const char *getDesc()
772 {
773 return mszDesc;
774 }
775 const char *getUdi()
776 {
777 return mszUdi;
778 }
779 const char *getNode()
780 {
781 return mszNode;
782 }
783};
784
785/**
786 * Helper function to query the sysfs subsystem for information about DVD
787 * drives attached to the system.
788 * @returns iprt status code
789 * @param pList where to add information about the drives detected
790 * @param isDVD are we looking for DVDs or floppies?
791 * @param pfSuccess Did we find anything?
792 *
793 * @returns IPRT status code
794 */
795/* static */
796int getDriveInfoFromSysfs(DriveInfoList *pList, bool isDVD, bool *pfSuccess)
797{
798 AssertPtrReturn(pList, VERR_INVALID_POINTER);
799 AssertPtrNullReturn(pfSuccess, VERR_INVALID_POINTER); /* Valid or Null */
800 LogFlowFunc (("pList=%p, isDVD=%u, pfSuccess=%p\n",
801 pList, (unsigned) isDVD, pfSuccess));
802 PRTDIR pDir = NULL;
803 RTDIRENTRY entry = {0};
804 int rc;
805 bool fSuccess = false;
806 unsigned cFound = 0;
807
808 if (!RTPathExists("/sys"))
809 return VINF_SUCCESS;
810 rc = RTDirOpen(&pDir, "/sys/block");
811 /* This might mean that sysfs semantics have changed */
812 AssertReturn(rc != VERR_FILE_NOT_FOUND, VINF_SUCCESS);
813 fSuccess = true;
814 if (RT_SUCCESS(rc))
815 while (true)
816 {
817 rc = RTDirRead(pDir, &entry, NULL);
818 Assert(rc != VERR_BUFFER_OVERFLOW); /* Should never happen... */
819 if (RT_FAILURE(rc)) /* Including overflow and no more files */
820 break;
821 if (entry.szName[0] == '.')
822 continue;
823 sysfsBlockDev dev(entry.szName, isDVD);
824 /* This might mean that sysfs semantics have changed */
825 AssertBreakStmt(dev.isConsistent(), fSuccess = false);
826 if (!dev.isValid())
827 continue;
828 try
829 {
830 pList->push_back(DriveInfo(dev.getNode(), dev.getUdi(),
831 dev.getDesc()));
832 }
833 catch(std::bad_alloc &e)
834 {
835 rc = VERR_NO_MEMORY;
836 break;
837 }
838 ++cFound;
839 }
840 RTDirClose(pDir);
841 if (rc == VERR_NO_MORE_FILES)
842 rc = VINF_SUCCESS;
843 if (RT_FAILURE(rc))
844 /* Clean up again */
845 for (unsigned i = 0; i < cFound; ++i)
846 pList->pop_back();
847 if (pfSuccess)
848 *pfSuccess = fSuccess;
849 LogFlow (("rc=%Rrc, fSuccess=%u\n", rc, (unsigned) fSuccess));
850 return rc;
851}
852
853
854/** Structure for holding information about a drive we have found */
855struct deviceNodeInfo
856{
857 /** The device number */
858 dev_t Device;
859 /** The device node path */
860 char szPath[RTPATH_MAX];
861 /** The device description */
862 char szDesc[256];
863 /** The device UDI */
864 char szUdi[256];
865};
866
867/** The maximum number of devices we will search for. */
868enum { MAX_DEVICE_NODES = 8 };
869/** An array of MAX_DEVICE_NODES devices */
870typedef struct deviceNodeInfo deviceNodeArray[MAX_DEVICE_NODES];
871
872/**
873 * Recursive worker function to walk the /dev tree looking for DVD or floppy
874 * devices.
875 * @returns true if we have already found MAX_DEVICE_NODES devices, false
876 * otherwise
877 * @param pszPath the path to start recursing. The function can modify
878 * this string at and after the terminating zero
879 * @param cchPath the size of the buffer (not the string!) in @a pszPath
880 * @param aDevices where to fill in information about devices that we have
881 * found
882 * @param wantDVD are we looking for DVD devices (or floppies)?
883 */
884static bool devFindDeviceRecursive(char *pszPath, size_t cchPath,
885 deviceNodeArray aDevices, bool wantDVD)
886{
887 /*
888 * Check assumptions made by the code below.
889 */
890 size_t const cchBasePath = strlen(pszPath);
891 AssertReturn(cchBasePath < RTPATH_MAX - 10U, false);
892 AssertReturn(pszPath[cchBasePath - 1] != '/', false);
893
894 PRTDIR pDir;
895 if (RT_FAILURE(RTDirOpen(&pDir, pszPath)))
896 return false;
897 for (;;)
898 {
899 RTDIRENTRY Entry;
900 RTFSOBJINFO ObjInfo;
901 int rc = RTDirRead(pDir, &Entry, NULL);
902 if (RT_FAILURE(rc))
903 break;
904 if (Entry.enmType == RTDIRENTRYTYPE_UNKNOWN)
905 {
906 if (RT_FAILURE(RTPathQueryInfo(pszPath, &ObjInfo,
907 RTFSOBJATTRADD_UNIX)))
908 continue;
909 if (RTFS_IS_SYMLINK(ObjInfo.Attr.fMode))
910 continue;
911 }
912
913 if (Entry.enmType == RTDIRENTRYTYPE_SYMLINK)
914 continue;
915 pszPath[cchBasePath] = '\0';
916 if (RT_FAILURE(RTPathAppend(pszPath, cchPath, Entry.szName)))
917 break;
918
919 /* Do the matching. */
920 dev_t DevNode;
921 char szDesc[256], szUdi[256];
922 if (!devValidateDevice(pszPath, wantDVD, &DevNode, szDesc,
923 sizeof(szDesc), szUdi, sizeof(szUdi)))
924 continue;
925 unsigned i;
926 for (i = 0; i < MAX_DEVICE_NODES; ++i)
927 if (!aDevices[i].Device || (aDevices[i].Device == DevNode))
928 break;
929 AssertBreak(i < MAX_DEVICE_NODES);
930 if (aDevices[i].Device)
931 continue;
932 aDevices[i].Device = DevNode;
933 RTStrPrintf(aDevices[i].szPath, sizeof(aDevices[i].szPath),
934 "%s", pszPath);
935 AssertCompile(sizeof(aDevices[i].szDesc) == sizeof(szDesc));
936 strcpy(aDevices[i].szDesc, szDesc);
937 AssertCompile(sizeof(aDevices[i].szUdi) == sizeof(szUdi));
938 strcpy(aDevices[i].szUdi, szUdi);
939 if (i == MAX_DEVICE_NODES - 1)
940 break;
941 continue;
942
943 /* Recurse into subdirectories. */
944 if ( (Entry.enmType == RTDIRENTRYTYPE_UNKNOWN)
945 && !RTFS_IS_DIRECTORY(ObjInfo.Attr.fMode))
946 continue;
947 if (Entry.enmType != RTDIRENTRYTYPE_DIRECTORY)
948 continue;
949 if (Entry.szName[0] == '.')
950 continue;
951
952 if (devFindDeviceRecursive(pszPath, cchPath, aDevices, wantDVD))
953 break;
954 }
955 RTDirClose(pDir);
956 return aDevices[MAX_DEVICE_NODES - 1].Device ? true : false;
957}
958
959
960/**
961 * Recursively walk through the /dev tree and add any DVD or floppy drives we
962 * find and can access to our list. (If we can't access them we can't check
963 * whether or not they are really DVD or floppy drives).
964 * @note this is rather slow (a couple of seconds) for DVD probing on
965 * systems with a static /dev tree, as the current code tries to open
966 * any device node with a major/minor combination that could belong to
967 * a CD-ROM device, and opening a non-existent device can take a non.
968 * negligeable time on Linux. If it is ever necessary to improve this
969 * (static /dev trees are no longer very fashionable these days, and
970 * sysfs looks like it will be with us for a while), we could further
971 * reduce the number of device nodes we open by checking whether the
972 * driver is actually loaded in /proc/devices, and by counting the
973 * of currently attached SCSI CD-ROM devices in /proc/scsi/scsi (yes,
974 * there is a race, but it is probably not important for us).
975 * @returns iprt status code
976 * @param pList the list to append the drives found to
977 * @param isDVD are we looking for DVD drives or for floppies?
978 * @param pfSuccess this will be set to true if we found at least one drive
979 * and to false otherwise. Optional.
980 */
981/* static */
982int getDriveInfoFromDev(DriveInfoList *pList, bool isDVD, bool *pfSuccess)
983{
984 AssertPtrReturn(pList, VERR_INVALID_POINTER);
985 AssertPtrNullReturn(pfSuccess, VERR_INVALID_POINTER);
986 LogFlowFunc(("pList=%p, isDVD=%d, pfSuccess=%p\n", pList, isDVD,
987 pfSuccess));
988 int rc = VINF_SUCCESS;
989 bool success = false;
990
991 deviceNodeArray aDevices = { { 0 } };
992 char szPath[RTPATH_MAX] = "/dev";
993 devFindDeviceRecursive(szPath, sizeof(szPath), aDevices, isDVD);
994 try
995 {
996 for (unsigned i = 0; i < MAX_DEVICE_NODES; ++i)
997 {
998 if (aDevices[i].Device)
999 {
1000 pList->push_back(DriveInfo(aDevices[i].szPath,
1001 aDevices[i].szUdi, aDevices[i].szDesc));
1002 success = true;
1003 }
1004 }
1005 if (pfSuccess != NULL)
1006 *pfSuccess = success;
1007 }
1008 catch(std::bad_alloc &e)
1009 {
1010 rc = VERR_NO_MEMORY;
1011 }
1012 LogFlowFunc (("rc=%Rrc, success=%d\n", rc, success));
1013 return rc;
1014}
1015
1016
1017int VBoxMainUSBDeviceInfo::UpdateDevices ()
1018{
1019 LogFlowThisFunc(("entered\n"));
1020 int rc = VINF_SUCCESS;
1021 bool success = false; /* Have we succeeded in finding anything yet? */
1022 try
1023 {
1024 bool halSuccess = false;
1025 mDeviceList.clear();
1026#if defined(RT_OS_LINUX)
1027#ifdef VBOX_WITH_DBUS
1028 if ( RT_SUCCESS(rc)
1029 && RT_SUCCESS(RTDBusLoadLib())
1030 && (!success || testing()))
1031 rc = getUSBDeviceInfoFromHal(&mDeviceList, &halSuccess);
1032 /* Try the old API if the new one *succeeded* as only one of them will
1033 * pick up devices anyway. */
1034 if (RT_SUCCESS(rc) && halSuccess && (!success || testing()))
1035 rc = getOldUSBDeviceInfoFromHal(&mDeviceList, &halSuccess);
1036 if (!success)
1037 success = halSuccess;
1038#endif /* VBOX_WITH_DBUS defined */
1039#endif /* RT_OS_LINUX */
1040 }
1041 catch(std::bad_alloc &e)
1042 {
1043 rc = VERR_NO_MEMORY;
1044 }
1045 LogFlowThisFunc(("rc=%Rrc\n", rc));
1046 return rc;
1047}
1048
1049struct VBoxMainHotplugWaiter::Context
1050{
1051#if defined RT_OS_LINUX && defined VBOX_WITH_DBUS
1052 /** The connection to DBus */
1053 RTMemAutoPtr <DBusConnection, VBoxHalShutdownPrivate> mConnection;
1054 /** Semaphore which is set when a device is hotplugged and reset when
1055 * it is read. */
1056 volatile bool mTriggered;
1057 /** A flag to say that we wish to interrupt the current wait. */
1058 volatile bool mInterrupt;
1059 /** Constructor */
1060 Context() : mTriggered(false), mInterrupt(false) {}
1061#endif /* defined RT_OS_LINUX && defined VBOX_WITH_DBUS */
1062};
1063
1064/* This constructor sets up a private connection to the DBus daemon, connects
1065 * to the hal service and installs a filter which sets the mTriggered flag in
1066 * the Context structure when a device (not necessarily USB) is added or
1067 * removed. */
1068VBoxMainHotplugWaiter::VBoxMainHotplugWaiter ()
1069{
1070#if defined RT_OS_LINUX && defined VBOX_WITH_DBUS
1071 int rc = VINF_SUCCESS;
1072
1073 mContext = new Context;
1074 if (RT_SUCCESS(RTDBusLoadLib()))
1075 {
1076 for (unsigned i = 0; RT_SUCCESS(rc) && i < 5 && !mContext->mConnection; ++i)
1077 {
1078 rc = halInitPrivate (&mContext->mConnection);
1079 }
1080 if (!mContext->mConnection)
1081 rc = VERR_NOT_SUPPORTED;
1082 DBusMessage *pMessage;
1083 while ( RT_SUCCESS(rc)
1084 && (pMessage = dbus_connection_pop_message (mContext->mConnection.get())) != NULL)
1085 dbus_message_unref (pMessage); /* empty the message queue. */
1086 if ( RT_SUCCESS(rc)
1087 && !dbus_connection_add_filter (mContext->mConnection.get(),
1088 dbusFilterFunction,
1089 (void *) &mContext->mTriggered, NULL))
1090 rc = VERR_NO_MEMORY;
1091 if (RT_FAILURE(rc))
1092 mContext->mConnection.reset();
1093 }
1094#endif /* defined RT_OS_LINUX && defined VBOX_WITH_DBUS */
1095}
1096
1097/* Destructor */
1098VBoxMainHotplugWaiter::~VBoxMainHotplugWaiter ()
1099{
1100#if defined RT_OS_LINUX && defined VBOX_WITH_DBUS
1101 if (!!mContext->mConnection)
1102 dbus_connection_remove_filter (mContext->mConnection.get(), dbusFilterFunction,
1103 (void *) &mContext->mTriggered);
1104 delete mContext;
1105#endif /* defined RT_OS_LINUX && defined VBOX_WITH_DBUS */
1106}
1107
1108/* Currently this is implemented using a timed out wait on our private DBus
1109 * connection. Because the connection is private we don't have to worry about
1110 * blocking other users. */
1111int VBoxMainHotplugWaiter::Wait(RTMSINTERVAL cMillies)
1112{
1113 int rc = VINF_SUCCESS;
1114#if defined RT_OS_LINUX && defined VBOX_WITH_DBUS
1115 if (!mContext->mConnection)
1116 rc = VERR_NOT_SUPPORTED;
1117 bool connected = true;
1118 mContext->mTriggered = false;
1119 mContext->mInterrupt = false;
1120 unsigned cRealMillies;
1121 if (cMillies != RT_INDEFINITE_WAIT)
1122 cRealMillies = cMillies;
1123 else
1124 cRealMillies = DBUS_POLL_TIMEOUT;
1125 while ( RT_SUCCESS(rc) && connected && !mContext->mTriggered
1126 && !mContext->mInterrupt)
1127 {
1128 connected = dbus_connection_read_write_dispatch (mContext->mConnection.get(),
1129 cRealMillies);
1130 if (mContext->mInterrupt)
1131 LogFlowFunc(("wait loop interrupted\n"));
1132 if (cMillies != RT_INDEFINITE_WAIT)
1133 mContext->mInterrupt = true;
1134 }
1135 if (!connected)
1136 rc = VERR_TRY_AGAIN;
1137#else /* !(defined RT_OS_LINUX && defined VBOX_WITH_DBUS) */
1138 rc = VERR_NOT_IMPLEMENTED;
1139#endif /* !(defined RT_OS_LINUX && defined VBOX_WITH_DBUS) */
1140 return rc;
1141}
1142
1143/* Set a flag to tell the Wait not to resume next time it times out. */
1144void VBoxMainHotplugWaiter::Interrupt()
1145{
1146#if defined RT_OS_LINUX && defined VBOX_WITH_DBUS
1147 LogFlowFunc(("\n"));
1148 mContext->mInterrupt = true;
1149#endif /* defined RT_OS_LINUX && defined VBOX_WITH_DBUS */
1150}
1151
1152
1153#if defined(RT_OS_LINUX) && defined(VBOX_WITH_DBUS)
1154/** Wrapper class around DBusError for automatic cleanup */
1155class autoDBusError
1156{
1157 DBusError mError;
1158public:
1159 autoDBusError () { dbus_error_init (&mError); }
1160 ~autoDBusError ()
1161 {
1162 if (IsSet())
1163 dbus_error_free (&mError);
1164 }
1165 DBusError &get () { return mError; }
1166 bool IsSet ()
1167 {
1168 Assert((mError.name == NULL) == (mError.message == NULL));
1169 return (mError.name != NULL);
1170 }
1171 bool HasName (const char *pcszName)
1172 {
1173 Assert((mError.name == NULL) == (mError.message == NULL));
1174 return (RTStrCmp (mError.name, pcszName) == 0);
1175 }
1176 void FlowLog ()
1177 {
1178 if (IsSet ())
1179 LogFlow(("DBus error %s: %s\n", mError.name, mError.message));
1180 }
1181};
1182
1183/**
1184 * Helper function for setting up a connection to the DBus daemon and
1185 * registering with the hal service.
1186 *
1187 * @note If libdbus is being loaded at runtime then be sure to call
1188 * VBoxDBusCheckPresence before calling this.
1189 * @returns iprt status code
1190 * @param ppConnection where to store the connection handle
1191 */
1192/* static */
1193int halInit (RTMemAutoPtr <DBusConnection, VBoxHalShutdown> *pConnection)
1194{
1195 AssertReturn(VALID_PTR (pConnection), VERR_INVALID_POINTER);
1196 LogFlowFunc (("pConnection=%p\n", pConnection));
1197 int rc = VINF_SUCCESS;
1198 bool halSuccess = true;
1199 autoDBusError dbusError;
1200
1201 RTMemAutoPtr <DBusConnection, VBoxDBusConnectionUnref> dbusConnection;
1202 dbusConnection = dbus_bus_get (DBUS_BUS_SYSTEM, &dbusError.get());
1203 if (!dbusConnection)
1204 halSuccess = false;
1205 if (halSuccess)
1206 {
1207 dbus_connection_set_exit_on_disconnect (dbusConnection.get(), false);
1208 halSuccess = dbus_bus_name_has_owner (dbusConnection.get(),
1209 "org.freedesktop.Hal", &dbusError.get());
1210 }
1211 if (halSuccess)
1212 {
1213 dbus_bus_add_match (dbusConnection.get(),
1214 "type='signal',"
1215 "interface='org.freedesktop.Hal.Manager',"
1216 "sender='org.freedesktop.Hal',"
1217 "path='/org/freedesktop/Hal/Manager'",
1218 &dbusError.get());
1219 halSuccess = !dbusError.IsSet();
1220 }
1221 if (dbusError.HasName (DBUS_ERROR_NO_MEMORY))
1222 rc = VERR_NO_MEMORY;
1223 if (halSuccess)
1224 *pConnection = dbusConnection.release();
1225 LogFlowFunc(("rc=%Rrc, (*pConnection).get()=%p\n", rc, (*pConnection).get()));
1226 dbusError.FlowLog();
1227 return rc;
1228}
1229
1230/**
1231 * Helper function for setting up a private connection to the DBus daemon and
1232 * registering with the hal service. Private connections are considered
1233 * unsociable and should not be used unnecessarily (as per the DBus API docs).
1234 *
1235 * @note If libdbus is being loaded at runtime then be sure to call
1236 * VBoxDBusCheckPresence before calling this.
1237 * @returns iprt status code
1238 * @param pConnection where to store the connection handle
1239 */
1240/* static */
1241int halInitPrivate (RTMemAutoPtr <DBusConnection, VBoxHalShutdownPrivate> *pConnection)
1242{
1243 AssertReturn(VALID_PTR (pConnection), VERR_INVALID_POINTER);
1244 LogFlowFunc (("pConnection=%p\n", pConnection));
1245 int rc = VINF_SUCCESS;
1246 bool halSuccess = true;
1247 autoDBusError dbusError;
1248
1249 RTMemAutoPtr <DBusConnection, VBoxDBusConnectionCloseAndUnref> dbusConnection;
1250 dbusConnection = dbus_bus_get_private (DBUS_BUS_SYSTEM, &dbusError.get());
1251 if (!dbusConnection)
1252 halSuccess = false;
1253 if (halSuccess)
1254 {
1255 dbus_connection_set_exit_on_disconnect (dbusConnection.get(), false);
1256 halSuccess = dbus_bus_name_has_owner (dbusConnection.get(),
1257 "org.freedesktop.Hal", &dbusError.get());
1258 }
1259 if (halSuccess)
1260 {
1261 dbus_bus_add_match (dbusConnection.get(),
1262 "type='signal',"
1263 "interface='org.freedesktop.Hal.Manager',"
1264 "sender='org.freedesktop.Hal',"
1265 "path='/org/freedesktop/Hal/Manager'",
1266 &dbusError.get());
1267 halSuccess = !dbusError.IsSet();
1268 }
1269 if (dbusError.HasName (DBUS_ERROR_NO_MEMORY))
1270 rc = VERR_NO_MEMORY;
1271 if (halSuccess)
1272 *pConnection = dbusConnection.release();
1273 LogFlowFunc(("rc=%Rrc, (*pConnection).get()=%p\n", rc, (*pConnection).get()));
1274 dbusError.FlowLog();
1275 return rc;
1276}
1277
1278/**
1279 * Helper function for shutting down a connection to DBus and hal.
1280 * @param pConnection the connection handle
1281 */
1282/* extern */
1283void VBoxHalShutdown (DBusConnection *pConnection)
1284{
1285 AssertReturnVoid(VALID_PTR (pConnection));
1286 LogFlowFunc (("pConnection=%p\n", pConnection));
1287 autoDBusError dbusError;
1288
1289 dbus_bus_remove_match (pConnection,
1290 "type='signal',"
1291 "interface='org.freedesktop.Hal.Manager',"
1292 "sender='org.freedesktop.Hal',"
1293 "path='/org/freedesktop/Hal/Manager'",
1294 &dbusError.get());
1295 dbus_connection_unref (pConnection);
1296 LogFlowFunc(("returning\n"));
1297 dbusError.FlowLog();
1298}
1299
1300/**
1301 * Helper function for shutting down a private connection to DBus and hal.
1302 * @param pConnection the connection handle
1303 */
1304/* extern */
1305void VBoxHalShutdownPrivate (DBusConnection *pConnection)
1306{
1307 AssertReturnVoid(VALID_PTR (pConnection));
1308 LogFlowFunc (("pConnection=%p\n", pConnection));
1309 autoDBusError dbusError;
1310
1311 dbus_bus_remove_match (pConnection,
1312 "type='signal',"
1313 "interface='org.freedesktop.Hal.Manager',"
1314 "sender='org.freedesktop.Hal',"
1315 "path='/org/freedesktop/Hal/Manager'",
1316 &dbusError.get());
1317 dbus_connection_close (pConnection);
1318 dbus_connection_unref (pConnection);
1319 LogFlowFunc(("returning\n"));
1320 dbusError.FlowLog();
1321}
1322
1323/** Wrapper around dbus_connection_unref. We need this to use it as a real
1324 * function in auto pointers, as a function pointer won't wash here. */
1325/* extern */
1326void VBoxDBusConnectionUnref(DBusConnection *pConnection)
1327{
1328 dbus_connection_unref(pConnection);
1329}
1330
1331/**
1332 * This function closes and unrefs a private connection to dbus. It should
1333 * only be called once no-one else is referencing the connection.
1334 */
1335/* extern */
1336void VBoxDBusConnectionCloseAndUnref(DBusConnection *pConnection)
1337{
1338 dbus_connection_close(pConnection);
1339 dbus_connection_unref(pConnection);
1340}
1341
1342/** Wrapper around dbus_message_unref. We need this to use it as a real
1343 * function in auto pointers, as a function pointer won't wash here. */
1344/* extern */
1345void VBoxDBusMessageUnref(DBusMessage *pMessage)
1346{
1347 dbus_message_unref(pMessage);
1348}
1349
1350/**
1351 * Find the UDIs of hal entries that contain Key=Value property.
1352 * @returns iprt status code. If a non-fatal error occurs, we return success
1353 * but reset pMessage to NULL.
1354 * @param pConnection an initialised connection DBus
1355 * @param pszKey the property key
1356 * @param pszValue the property value
1357 * @param pMessage where to store the return DBus message. This must be
1358 * parsed to get at the UDIs. NOT optional.
1359 */
1360/* static */
1361int halFindDeviceStringMatch (DBusConnection *pConnection, const char *pszKey,
1362 const char *pszValue,
1363 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> *pMessage)
1364{
1365 AssertReturn( VALID_PTR (pConnection) && VALID_PTR (pszKey)
1366 && VALID_PTR (pszValue) && VALID_PTR (pMessage),
1367 VERR_INVALID_POINTER);
1368 LogFlowFunc (("pConnection=%p, pszKey=%s, pszValue=%s, pMessage=%p\n",
1369 pConnection, pszKey, pszValue, pMessage));
1370 int rc = VINF_SUCCESS; /* We set this to failure on fatal errors. */
1371 bool halSuccess = true; /* We set this to false to abort the operation. */
1372 autoDBusError dbusError;
1373
1374 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> message, reply;
1375 if (halSuccess && RT_SUCCESS(rc))
1376 {
1377 message = dbus_message_new_method_call ("org.freedesktop.Hal",
1378 "/org/freedesktop/Hal/Manager",
1379 "org.freedesktop.Hal.Manager",
1380 "FindDeviceStringMatch");
1381 if (!message)
1382 rc = VERR_NO_MEMORY;
1383 }
1384 if (halSuccess && RT_SUCCESS(rc))
1385 {
1386 DBusMessageIter iterAppend;
1387 dbus_message_iter_init_append (message.get(), &iterAppend);
1388 dbus_message_iter_append_basic (&iterAppend, DBUS_TYPE_STRING, &pszKey);
1389 dbus_message_iter_append_basic (&iterAppend, DBUS_TYPE_STRING, &pszValue);
1390 reply = dbus_connection_send_with_reply_and_block (pConnection,
1391 message.get(), -1,
1392 &dbusError.get());
1393 if (!reply)
1394 halSuccess = false;
1395 }
1396 *pMessage = reply.release ();
1397 LogFlowFunc (("rc=%Rrc, *pMessage.value()=%p\n", rc, (*pMessage).get()));
1398 dbusError.FlowLog();
1399 return rc;
1400}
1401
1402/**
1403 * Find the UDIs of hal entries that contain Key=Value property and return the
1404 * result on the end of a vector of iprt::MiniString.
1405 * @returns iprt status code. If a non-fatal error occurs, we return success
1406 * but set *pfSuccess to false.
1407 * @param pConnection an initialised connection DBus
1408 * @param pszKey the property key
1409 * @param pszValue the property value
1410 * @param pMatches pointer to an array of iprt::MiniString to append the
1411 * results to. NOT optional.
1412 * @param pfSuccess will be set to true if the operation succeeds
1413 */
1414/* static */
1415int halFindDeviceStringMatchVector (DBusConnection *pConnection,
1416 const char *pszKey, const char *pszValue,
1417 std::vector<iprt::MiniString> *pMatches,
1418 bool *pfSuccess)
1419{
1420 AssertPtrReturn (pConnection, VERR_INVALID_POINTER);
1421 AssertPtrReturn (pszKey, VERR_INVALID_POINTER);
1422 AssertPtrReturn (pszValue, VERR_INVALID_POINTER);
1423 AssertPtrReturn (pMatches, VERR_INVALID_POINTER);
1424 AssertReturn(pfSuccess == NULL || VALID_PTR (pfSuccess), VERR_INVALID_POINTER);
1425 LogFlowFunc (("pConnection=%p, pszKey=%s, pszValue=%s, pMatches=%p, pfSuccess=%p\n",
1426 pConnection, pszKey, pszValue, pMatches, pfSuccess));
1427 int rc = VINF_SUCCESS; /* We set this to failure on fatal errors. */
1428 bool halSuccess = true; /* We set this to false to abort the operation. */
1429
1430 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> message, replyFind;
1431 DBusMessageIter iterFind, iterUdis;
1432
1433 if (halSuccess && RT_SUCCESS(rc))
1434 {
1435 rc = halFindDeviceStringMatch (pConnection, pszKey, pszValue,
1436 &replyFind);
1437 if (!replyFind)
1438 halSuccess = false;
1439 }
1440 if (halSuccess && RT_SUCCESS(rc))
1441 {
1442 dbus_message_iter_init (replyFind.get(), &iterFind);
1443 if (dbus_message_iter_get_arg_type (&iterFind) != DBUS_TYPE_ARRAY)
1444 halSuccess = false;
1445 }
1446 if (halSuccess && RT_SUCCESS(rc))
1447 dbus_message_iter_recurse (&iterFind, &iterUdis);
1448 for (; halSuccess && RT_SUCCESS(rc)
1449 && dbus_message_iter_get_arg_type (&iterUdis) == DBUS_TYPE_STRING;
1450 dbus_message_iter_next(&iterUdis))
1451 {
1452 /* Now get all UDIs from the iterator */
1453 const char *pszUdi;
1454 dbus_message_iter_get_basic (&iterUdis, &pszUdi);
1455 try
1456 {
1457 pMatches->push_back(pszUdi);
1458 }
1459 catch(std::bad_alloc &e)
1460 {
1461 rc = VERR_NO_MEMORY;
1462 }
1463 }
1464 if (pfSuccess != NULL)
1465 *pfSuccess = halSuccess;
1466 LogFlow (("rc=%Rrc, halSuccess=%d\n", rc, halSuccess));
1467 return rc;
1468}
1469
1470/**
1471 * Read a set of string properties for a device. If some of the properties are
1472 * not of type DBUS_TYPE_STRING or do not exist then a NULL pointer will be
1473 * returned for them.
1474 * @returns iprt status code. If the operation failed for non-fatal reasons
1475 * then we return success and leave pMessage untouched - reset it
1476 * before the call to detect this.
1477 * @param pConnection an initialised connection DBus
1478 * @param pszUdi the Udi of the device
1479 * @param cProps the number of property values to look up
1480 * @param papszKeys the keys of the properties to be looked up
1481 * @param papszValues where to store the values of the properties. The
1482 * strings returned will be valid until the message
1483 * returned in @a ppMessage is freed. Undefined if
1484 * the message is NULL.
1485 * @param pMessage where to store the return DBus message. The caller
1486 * is responsible for freeing this once they have
1487 * finished with the value strings. NOT optional.
1488 */
1489/* static */
1490int halGetPropertyStrings (DBusConnection *pConnection, const char *pszUdi,
1491 size_t cProps, const char **papszKeys,
1492 char **papszValues,
1493 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> *pMessage)
1494{
1495 AssertReturn( VALID_PTR (pConnection) && VALID_PTR (pszUdi)
1496 && VALID_PTR (papszKeys) && VALID_PTR (papszValues)
1497 && VALID_PTR (pMessage),
1498 VERR_INVALID_POINTER);
1499 LogFlowFunc (("pConnection=%p, pszUdi=%s, cProps=%llu, papszKeys=%p, papszValues=%p, pMessage=%p\n",
1500 pConnection, pszUdi, cProps, papszKeys, papszValues, pMessage));
1501 int rc = VINF_SUCCESS; /* We set this to failure on fatal errors. */
1502 bool halSuccess = true; /* We set this to false to abort the operation. */
1503 autoDBusError dbusError;
1504
1505 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> message, reply;
1506 DBusMessageIter iterGet, iterProps;
1507
1508 /* Initialise the return array to NULLs */
1509 for (size_t i = 0; i < cProps; ++i)
1510 papszValues[i] = NULL;
1511
1512 /* Send a GetAllProperties message to hald */
1513 message = dbus_message_new_method_call ("org.freedesktop.Hal", pszUdi,
1514 "org.freedesktop.Hal.Device",
1515 "GetAllProperties");
1516 if (!message)
1517 rc = VERR_NO_MEMORY;
1518 if (halSuccess && RT_SUCCESS(rc))
1519 {
1520 reply = dbus_connection_send_with_reply_and_block (pConnection,
1521 message.get(), -1,
1522 &dbusError.get());
1523 if (!reply)
1524 halSuccess = false;
1525 }
1526
1527 /* Parse the reply */
1528 if (halSuccess && RT_SUCCESS(rc))
1529 {
1530 dbus_message_iter_init (reply.get(), &iterGet);
1531 if ( dbus_message_iter_get_arg_type (&iterGet) != DBUS_TYPE_ARRAY
1532 && dbus_message_iter_get_element_type (&iterGet) != DBUS_TYPE_DICT_ENTRY)
1533 halSuccess = false;
1534 }
1535 if (halSuccess && RT_SUCCESS(rc))
1536 dbus_message_iter_recurse (&iterGet, &iterProps);
1537 /* Go through all entries in the reply and see if any match our keys. */
1538 while ( halSuccess && RT_SUCCESS(rc)
1539 && dbus_message_iter_get_arg_type (&iterProps)
1540 == DBUS_TYPE_DICT_ENTRY)
1541 {
1542 const char *pszKey;
1543 DBusMessageIter iterEntry, iterValue;
1544 dbus_message_iter_recurse (&iterProps, &iterEntry);
1545 dbus_message_iter_get_basic (&iterEntry, &pszKey);
1546 dbus_message_iter_next (&iterEntry);
1547 dbus_message_iter_recurse (&iterEntry, &iterValue);
1548 /* Fill in any matches. */
1549 for (size_t i = 0; i < cProps; ++i)
1550 if (strcmp (pszKey, papszKeys[i]) == 0)
1551 {
1552 if (dbus_message_iter_get_arg_type (&iterValue) == DBUS_TYPE_STRING)
1553 dbus_message_iter_get_basic (&iterValue, &papszValues[i]);
1554 }
1555 dbus_message_iter_next (&iterProps);
1556 }
1557 if (RT_SUCCESS(rc) && halSuccess)
1558 *pMessage = reply.release();
1559 if (dbusError.HasName (DBUS_ERROR_NO_MEMORY))
1560 rc = VERR_NO_MEMORY;
1561 LogFlowFunc (("rc=%Rrc, *pMessage.value()=%p\n", rc, (*pMessage).get()));
1562 dbusError.FlowLog();
1563 return rc;
1564}
1565
1566/**
1567 * Read a set of string properties for a device. If some properties do not
1568 * exist or are not of type DBUS_TYPE_STRING, we will still fetch the others.
1569 * @returns iprt status code. If the operation failed for non-fatal reasons
1570 * then we return success and set *pfSuccess to false.
1571 * @param pConnection an initialised connection DBus
1572 * @param pszUdi the Udi of the device
1573 * @param cProps the number of property values to look up
1574 * @param papszKeys the keys of the properties to be looked up
1575 * @param pMatches pointer to an empty array of iprt::MiniString to append the
1576 * results to. NOT optional.
1577 * @param pfMatches pointer to an array of boolean values indicating
1578 * whether the respective property is a string. If this
1579 * is not supplied then all properties must be strings
1580 * for the operation to be considered successful
1581 * @param pfSuccess will be set to true if the operation succeeds
1582 */
1583/* static */
1584int halGetPropertyStringsVector (DBusConnection *pConnection,
1585 const char *pszUdi, size_t cProps,
1586 const char **papszKeys,
1587 std::vector<iprt::MiniString> *pMatches,
1588 bool *pfMatches, bool *pfSuccess)
1589{
1590 AssertPtrReturn (pConnection, VERR_INVALID_POINTER);
1591 AssertPtrReturn (pszUdi, VERR_INVALID_POINTER);
1592 AssertPtrReturn (papszKeys, VERR_INVALID_POINTER);
1593 AssertPtrReturn (pMatches, VERR_INVALID_POINTER);
1594 AssertReturn((pfMatches == NULL) || VALID_PTR (pfMatches), VERR_INVALID_POINTER);
1595 AssertReturn((pfSuccess == NULL) || VALID_PTR (pfSuccess), VERR_INVALID_POINTER);
1596 AssertReturn(pMatches->empty(), VERR_INVALID_PARAMETER);
1597 LogFlowFunc (("pConnection=%p, pszUdi=%s, cProps=%llu, papszKeys=%p, pMatches=%p, pfMatches=%p, pfSuccess=%p\n",
1598 pConnection, pszUdi, cProps, papszKeys, pMatches, pfMatches, pfSuccess));
1599 RTMemAutoPtr <char *> values(cProps);
1600 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> message;
1601 bool halSuccess = true;
1602 int rc = halGetPropertyStrings (pConnection, pszUdi, cProps, papszKeys,
1603 values.get(), &message);
1604 if (!message)
1605 halSuccess = false;
1606 for (size_t i = 0; RT_SUCCESS(rc) && halSuccess && i < cProps; ++i)
1607 {
1608 bool fMatches = values[i] != NULL;
1609 if (pfMatches != NULL)
1610 pfMatches[i] = fMatches;
1611 else
1612 halSuccess = fMatches;
1613 try
1614 {
1615 pMatches->push_back(fMatches ? values[i] : "");
1616 }
1617 catch(std::bad_alloc &e)
1618 {
1619 rc = VERR_NO_MEMORY;
1620 }
1621 }
1622 if (pfSuccess != NULL)
1623 *pfSuccess = halSuccess;
1624 if (RT_SUCCESS(rc) && halSuccess)
1625 {
1626 Assert(pMatches->size() == cProps);
1627 AssertForEach(j, size_t, 0, cProps, (pfMatches == NULL)
1628 || (pfMatches[j] == true)
1629 || ((pfMatches[j] == false) && (pMatches[j].size() == 0)));
1630 }
1631 LogFlowFunc (("rc=%Rrc, halSuccess=%d\n", rc, halSuccess));
1632 return rc;
1633}
1634
1635
1636/**
1637 * Helper function to query the hal subsystem for information about USB devices
1638 * attached to the system.
1639 * @returns iprt status code
1640 * @param pList where to add information about the devices detected
1641 * @param pfSuccess will be set to true if all interactions with hal
1642 * succeeded and to false otherwise. Optional.
1643 *
1644 * @returns IPRT status code
1645 */
1646/* static */
1647int getUSBDeviceInfoFromHal(USBDeviceInfoList *pList, bool *pfSuccess)
1648{
1649 AssertReturn(VALID_PTR (pList) && (pfSuccess == NULL || VALID_PTR (pfSuccess)),
1650 VERR_INVALID_POINTER);
1651 LogFlowFunc (("pList=%p, pfSuccess=%p\n", pList, pfSuccess));
1652 int rc = VINF_SUCCESS; /* We set this to failure on fatal errors. */
1653 bool halSuccess = true; /* We set this to false to abort the operation. */
1654 autoDBusError dbusError;
1655
1656 RTMemAutoPtr<DBusMessage, VBoxDBusMessageUnref> message, replyFind, replyGet;
1657 RTMemAutoPtr<DBusConnection, VBoxHalShutdown> dbusConnection;
1658 DBusMessageIter iterFind, iterUdis;
1659
1660 /* Connect to hal */
1661 rc = halInit (&dbusConnection);
1662 if (!dbusConnection)
1663 halSuccess = false;
1664 /* Get an array of all devices in the usb_device subsystem */
1665 if (halSuccess && RT_SUCCESS(rc))
1666 {
1667 rc = halFindDeviceStringMatch(dbusConnection.get(), "info.subsystem",
1668 "usb_device", &replyFind);
1669 if (!replyFind)
1670 halSuccess = false;
1671 }
1672 if (halSuccess && RT_SUCCESS(rc))
1673 {
1674 dbus_message_iter_init(replyFind.get(), &iterFind);
1675 if (dbus_message_iter_get_arg_type (&iterFind) != DBUS_TYPE_ARRAY)
1676 halSuccess = false;
1677 }
1678 /* Recurse down into the array and query interesting information about the
1679 * entries. */
1680 if (halSuccess && RT_SUCCESS(rc))
1681 dbus_message_iter_recurse(&iterFind, &iterUdis);
1682 for (; halSuccess && RT_SUCCESS(rc)
1683 && dbus_message_iter_get_arg_type(&iterUdis) == DBUS_TYPE_STRING;
1684 dbus_message_iter_next(&iterUdis))
1685 {
1686 /* Get the device node and the sysfs path for the current entry. */
1687 const char *pszUdi;
1688 dbus_message_iter_get_basic (&iterUdis, &pszUdi);
1689 static const char *papszKeys[] = { "linux.device_file", "linux.sysfs_path" };
1690 char *papszValues[RT_ELEMENTS(papszKeys)];
1691 rc = halGetPropertyStrings(dbusConnection.get(), pszUdi, RT_ELEMENTS(papszKeys),
1692 papszKeys, papszValues, &replyGet);
1693 const char *pszDevice = papszValues[0], *pszSysfsPath = papszValues[1];
1694 /* Get the interfaces. */
1695 if (!!replyGet && pszDevice && pszSysfsPath)
1696 {
1697 USBDeviceInfo info(pszDevice, pszSysfsPath);
1698 bool ifaceSuccess = true; /* If we can't get the interfaces, just
1699 * skip this one device. */
1700 rc = getUSBInterfacesFromHal(&info.mInterfaces, pszUdi, &ifaceSuccess);
1701 if (RT_SUCCESS(rc) && halSuccess && ifaceSuccess)
1702 try
1703 {
1704 pList->push_back(info);
1705 }
1706 catch(std::bad_alloc &e)
1707 {
1708 rc = VERR_NO_MEMORY;
1709 }
1710 }
1711 }
1712 if (dbusError.HasName (DBUS_ERROR_NO_MEMORY))
1713 rc = VERR_NO_MEMORY;
1714 if (pfSuccess != NULL)
1715 *pfSuccess = halSuccess;
1716 LogFlow(("rc=%Rrc, halSuccess=%d\n", rc, halSuccess));
1717 dbusError.FlowLog();
1718 return rc;
1719}
1720
1721/**
1722 * Helper function to query the hal subsystem for information about USB devices
1723 * attached to the system, using the older API.
1724 * @returns iprt status code
1725 * @param pList where to add information about the devices detected
1726 * @param pfSuccess will be set to true if all interactions with hal
1727 * succeeded and to false otherwise. Optional.
1728 *
1729 * @returns IPRT status code
1730 */
1731/* static */
1732int getOldUSBDeviceInfoFromHal(USBDeviceInfoList *pList, bool *pfSuccess)
1733{
1734 AssertReturn(VALID_PTR (pList) && (pfSuccess == NULL || VALID_PTR (pfSuccess)),
1735 VERR_INVALID_POINTER);
1736 LogFlowFunc (("pList=%p, pfSuccess=%p\n", pList, pfSuccess));
1737 int rc = VINF_SUCCESS; /* We set this to failure on fatal errors. */
1738 bool halSuccess = true; /* We set this to false to abort the operation. */
1739 autoDBusError dbusError;
1740
1741 RTMemAutoPtr<DBusMessage, VBoxDBusMessageUnref> message, replyFind, replyGet;
1742 RTMemAutoPtr<DBusConnection, VBoxHalShutdown> dbusConnection;
1743 DBusMessageIter iterFind, iterUdis;
1744
1745 /* Connect to hal */
1746 rc = halInit(&dbusConnection);
1747 if (!dbusConnection)
1748 halSuccess = false;
1749 /* Get an array of all devices in the usb_device subsystem */
1750 if (halSuccess && RT_SUCCESS(rc))
1751 {
1752 rc = halFindDeviceStringMatch(dbusConnection.get(), "info.category",
1753 "usbraw", &replyFind);
1754 if (!replyFind)
1755 halSuccess = false;
1756 }
1757 if (halSuccess && RT_SUCCESS(rc))
1758 {
1759 dbus_message_iter_init(replyFind.get(), &iterFind);
1760 if (dbus_message_iter_get_arg_type(&iterFind) != DBUS_TYPE_ARRAY)
1761 halSuccess = false;
1762 }
1763 /* Recurse down into the array and query interesting information about the
1764 * entries. */
1765 if (halSuccess && RT_SUCCESS(rc))
1766 dbus_message_iter_recurse(&iterFind, &iterUdis);
1767 for (; halSuccess && RT_SUCCESS(rc)
1768 && dbus_message_iter_get_arg_type(&iterUdis) == DBUS_TYPE_STRING;
1769 dbus_message_iter_next(&iterUdis))
1770 {
1771 /* Get the device node and the sysfs path for the current entry. */
1772 const char *pszUdi;
1773 dbus_message_iter_get_basic(&iterUdis, &pszUdi);
1774 static const char *papszKeys[] = { "linux.device_file", "info.parent" };
1775 char *papszValues[RT_ELEMENTS(papszKeys)];
1776 rc = halGetPropertyStrings(dbusConnection.get(), pszUdi, RT_ELEMENTS(papszKeys),
1777 papszKeys, papszValues, &replyGet);
1778 const char *pszDevice = papszValues[0], *pszSysfsPath = papszValues[1];
1779 /* Get the interfaces. */
1780 if (!!replyGet && pszDevice && pszSysfsPath)
1781 {
1782 USBDeviceInfo info(pszDevice, pszSysfsPath);
1783 bool ifaceSuccess = false; /* If we can't get the interfaces, just
1784 * skip this one device. */
1785 rc = getUSBInterfacesFromHal(&info.mInterfaces, pszSysfsPath,
1786 &ifaceSuccess);
1787 if (RT_SUCCESS(rc) && halSuccess && ifaceSuccess)
1788 try
1789 {
1790 pList->push_back(info);
1791 }
1792 catch(std::bad_alloc &e)
1793 {
1794 rc = VERR_NO_MEMORY;
1795 }
1796 }
1797 }
1798 if (dbusError.HasName(DBUS_ERROR_NO_MEMORY))
1799 rc = VERR_NO_MEMORY;
1800 if (pfSuccess != NULL)
1801 *pfSuccess = halSuccess;
1802 LogFlow(("rc=%Rrc, halSuccess=%d\n", rc, halSuccess));
1803 dbusError.FlowLog();
1804 return rc;
1805}
1806
1807/**
1808 * Helper function to query the hal subsystem for information about USB devices
1809 * attached to the system.
1810 * @returns iprt status code
1811 * @param pList where to add information about the devices detected. If
1812 * certain interfaces are not found (@a pfFound is false on
1813 * return) this may contain invalid information.
1814 * @param pcszUdi the hal UDI of the device
1815 * @param pfSuccess will be set to true if the operation succeeds and to
1816 * false if it fails for non-critical reasons. Optional.
1817 *
1818 * @returns IPRT status code
1819 */
1820/* static */
1821int getUSBInterfacesFromHal(std::vector<iprt::MiniString> *pList,
1822 const char *pcszUdi, bool *pfSuccess)
1823{
1824 AssertReturn(VALID_PTR(pList) && VALID_PTR(pcszUdi) &&
1825 (pfSuccess == NULL || VALID_PTR (pfSuccess)),
1826 VERR_INVALID_POINTER);
1827 LogFlowFunc(("pList=%p, pcszUdi=%s, pfSuccess=%p\n", pList, pcszUdi,
1828 pfSuccess));
1829 int rc = VINF_SUCCESS; /* We set this to failure on fatal errors. */
1830 bool halSuccess = true; /* We set this to false to abort the operation. */
1831 autoDBusError dbusError;
1832
1833 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> message, replyFind, replyGet;
1834 RTMemAutoPtr <DBusConnection, VBoxHalShutdown> dbusConnection;
1835 DBusMessageIter iterFind, iterUdis;
1836
1837 rc = halInit(&dbusConnection);
1838 if (!dbusConnection)
1839 halSuccess = false;
1840 if (halSuccess && RT_SUCCESS(rc))
1841 {
1842 /* Look for children of the current UDI. */
1843 rc = halFindDeviceStringMatch(dbusConnection.get(), "info.parent",
1844 pcszUdi, &replyFind);
1845 if (!replyFind)
1846 halSuccess = false;
1847 }
1848 if (halSuccess && RT_SUCCESS(rc))
1849 {
1850 dbus_message_iter_init(replyFind.get(), &iterFind);
1851 if (dbus_message_iter_get_arg_type(&iterFind) != DBUS_TYPE_ARRAY)
1852 halSuccess = false;
1853 }
1854 if (halSuccess && RT_SUCCESS(rc))
1855 dbus_message_iter_recurse(&iterFind, &iterUdis);
1856 for (; halSuccess && RT_SUCCESS(rc)
1857 && dbus_message_iter_get_arg_type(&iterUdis) == DBUS_TYPE_STRING;
1858 dbus_message_iter_next(&iterUdis))
1859 {
1860 /* Now get the sysfs path and the subsystem from the iterator */
1861 const char *pszUdi;
1862 dbus_message_iter_get_basic(&iterUdis, &pszUdi);
1863 static const char *papszKeys[] = { "linux.sysfs_path", "info.subsystem",
1864 "linux.subsystem" };
1865 char *papszValues[RT_ELEMENTS(papszKeys)];
1866 rc = halGetPropertyStrings(dbusConnection.get(), pszUdi, RT_ELEMENTS(papszKeys),
1867 papszKeys, papszValues, &replyGet);
1868 const char *pszSysfsPath = papszValues[0], *pszInfoSubsystem = papszValues[1],
1869 *pszLinuxSubsystem = papszValues[2];
1870 if (!replyGet)
1871 halSuccess = false;
1872 if (!!replyGet && pszSysfsPath == NULL)
1873 halSuccess = false;
1874 if ( halSuccess && RT_SUCCESS(rc)
1875 && RTStrCmp (pszInfoSubsystem, "usb_device") != 0 /* Children of buses can also be devices. */
1876 && RTStrCmp (pszLinuxSubsystem, "usb_device") != 0)
1877 try
1878 {
1879 pList->push_back(pszSysfsPath);
1880 }
1881 catch(std::bad_alloc &e)
1882 {
1883 rc = VERR_NO_MEMORY;
1884 }
1885 }
1886 if (dbusError.HasName(DBUS_ERROR_NO_MEMORY))
1887 rc = VERR_NO_MEMORY;
1888 if (pfSuccess != NULL)
1889 *pfSuccess = halSuccess;
1890 LogFlow(("rc=%Rrc, halSuccess=%d\n", rc, halSuccess));
1891 dbusError.FlowLog();
1892 return rc;
1893}
1894
1895/**
1896 * When it is registered with DBus, this function will be called by
1897 * dbus_connection_read_write_dispatch each time a message is received over the
1898 * DBus connection. We check whether that message was caused by a hal device
1899 * hotplug event, and if so we set a flag. dbus_connection_read_write_dispatch
1900 * will return after calling its filter functions, and its caller should then
1901 * check the status of the flag passed to the filter function.
1902 *
1903 * @param pConnection The DBus connection we are using.
1904 * @param pMessage The DBus message which just arrived.
1905 * @param pvUser A pointer to the flag variable we are to set.
1906 */
1907/* static */
1908DBusHandlerResult dbusFilterFunction(DBusConnection * /* pConnection */,
1909 DBusMessage *pMessage, void *pvUser)
1910{
1911 volatile bool *pTriggered = reinterpret_cast<volatile bool *>(pvUser);
1912 if ( dbus_message_is_signal(pMessage, "org.freedesktop.Hal.Manager",
1913 "DeviceAdded")
1914 || dbus_message_is_signal(pMessage, "org.freedesktop.Hal.Manager",
1915 "DeviceRemoved"))
1916 {
1917 *pTriggered = true;
1918 }
1919 return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
1920}
1921#endif /* RT_OS_LINUX && VBOX_WITH_DBUS */
1922
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