VirtualBox

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

Last change on this file since 15239 was 15058, checked in by vboxsync, 16 years ago

Main: fix a burn

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 26.1 KB
Line 
1/* $Id: HostHardwareLinux.cpp 15058 2008-12-05 22:11:21Z 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) 2006-2007 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
34#include <iprt/env.h>
35#include <iprt/mem.h>
36#include <iprt/string.h>
37
38#ifdef RT_OS_LINUX
39# include <sys/types.h>
40# include <sys/stat.h>
41# include <unistd.h>
42# include <sys/ioctl.h>
43# include <fcntl.h>
44# include <mntent.h>
45/* bird: This is a hack to work around conflicts between these linux kernel headers
46 * and the GLIBC tcpip headers. They have different declarations of the 4
47 * standard byte order functions. */
48// # define _LINUX_BYTEORDER_GENERIC_H
49# include <linux/cdrom.h>
50# ifdef VBOX_WITH_DBUS
51# include <vbox-dbus.h>
52# endif
53# include <errno.h>
54#endif /* RT_OS_LINUX */
55
56/*******************************************************************************
57* Global Variables *
58*******************************************************************************/
59
60bool g_testHostHardwareLinux = false;
61static bool testing () { return g_testHostHardwareLinux; }
62
63/*******************************************************************************
64* Defines and Typedefs *
65*******************************************************************************/
66
67typedef VBoxMainDriveInfo::DriveInfoList DriveInfoList;
68typedef VBoxMainDriveInfo::DriveInfo DriveInfo;
69
70static bool validateDevice(const char *deviceNode, bool isDVD);
71static int getDriveInfoFromEnv(const char *pszVar, DriveInfoList *pList,
72 bool isDVD, bool *pfSuccess);
73static int getDVDInfoFromMTab(char *mountTable, DriveInfoList *pList);
74#ifdef VBOX_WITH_DBUS
75static int halInit(DBusConnection **ppConnection);
76/* This must be extern to be used in the RTMemAutoPtr template */
77extern void halShutdown (DBusConnection *pConnection);
78static int halFindDeviceStringMatch (DBusConnection *pConnection,
79 const char *pszKey, const char *pszValue,
80 DBusMessage **ppMessage);
81static int halGetPropertyStrings (DBusConnection *pConnection,
82 const char *pszUdi, size_t cKeys,
83 const char **papszKeys, char **papszValues,
84 DBusMessage **ppMessage);
85static int getDriveInfoFromHal(DriveInfoList *pList, bool isDVD,
86 bool *pfSuccess);
87#endif /* VBOX_WITH_DBUS */
88
89/**
90 * Updates the list of host DVD drives.
91 *
92 * @returns iprt status code
93 */
94int VBoxMainDriveInfo::updateDVDs ()
95{
96 LogFlowThisFunc (("entered\n"));
97 int rc = VINF_SUCCESS;
98 bool success = false; /* Have we succeeded in finding anything yet? */
99 try
100 {
101 mDVDList.clear ();
102#if defined(RT_OS_LINUX)
103#ifdef VBOX_WITH_DBUS
104 if (RT_SUCCESS (rc) && VBoxDBusCheckPresence() && (!success || testing()))
105 rc = getDriveInfoFromHal(&mDVDList, true /* isDVD */, &success);
106#endif /* VBOX_WITH_DBUS defined */
107 // On Linux without hal, the situation is much more complex. We will take a
108 // heuristical approach and also allow the user to specify a list of host
109 // CDROMs using an environment variable.
110 // The general strategy is to try some known device names and see of they
111 // exist. At last, we'll enumerate the /etc/fstab file (luckily there's an
112 // API to parse it) for CDROM devices. Ok, let's start!
113 if (RT_SUCCESS (rc) && (!success || testing()))
114 rc = getDriveInfoFromEnv ("VBOX_CDROM", &mDVDList, true /* isDVD */,
115 &success);
116 if (RT_SUCCESS (rc) && (!success || testing()))
117 {
118 // this is a good guess usually
119 if (validateDevice("/dev/cdrom", true))
120 mDVDList.push_back (DriveInfo ("/dev/cdrom"));
121
122 // check the mounted drives
123 rc = getDVDInfoFromMTab((char*)"/etc/mtab", &mDVDList);
124
125 // check the drives that can be mounted
126 if (RT_SUCCESS (rc))
127 rc = getDVDInfoFromMTab((char*)"/etc/fstab", &mDVDList);
128 }
129#endif
130 }
131 catch (std::bad_alloc)
132 {
133 rc = VERR_NO_MEMORY;
134 }
135 LogFlowThisFunc (("rc=%Rrc\n", rc));
136 return rc;
137}
138
139/**
140 * Updates the list of host floppy drives.
141 *
142 * @returns iprt status code
143 */
144int VBoxMainDriveInfo::updateFloppies ()
145{
146 LogFlowThisFunc (("entered\n"));
147 int rc = VINF_SUCCESS;
148 bool success = false; /* Have we succeeded in finding anything yet? */
149 try
150 {
151 mFloppyList.clear ();
152#if defined(RT_OS_LINUX)
153#ifdef VBOX_WITH_DBUS
154 if (RT_SUCCESS (rc) && VBoxDBusCheckPresence() && (!success || testing()))
155 rc = getDriveInfoFromHal(&mFloppyList, false /* isDVD */, &success);
156#endif /* VBOX_WITH_DBUS defined */
157 // As with the CDROMs, on Linux we have to take a multi-level approach
158 // involving parsing the mount tables. As this is not bulletproof, we'll
159 // give the user the chance to override the detection by an environment
160 // variable and skip the detection.
161 if (RT_SUCCESS (rc) && (!success || testing()))
162 rc = getDriveInfoFromEnv ("VBOX_FLOPPY", &mFloppyList, false /* isDVD */,
163 &success);
164
165 if (RT_SUCCESS (rc) && (!success || testing()))
166 {
167 // we assume that a floppy is always /dev/fd[x] with x from 0 to 7
168 char devName[10];
169 for (int i = 0; i <= 7; i++)
170 {
171 sprintf(devName, "/dev/fd%d", i);
172 if (validateDevice(devName, false))
173 mFloppyList.push_back (DriveInfo (devName));
174 }
175 }
176#endif
177 }
178 catch (std::bad_alloc)
179 {
180 rc = VERR_NO_MEMORY;
181 }
182 LogFlowThisFunc (("rc=%Rrc\n", rc));
183 return rc;
184}
185
186#ifdef RT_OS_LINUX
187/**
188 * Helper function to check whether the given device node is a valid drive
189 */
190/* static */
191bool validateDevice(const char *deviceNode, bool isDVD)
192{
193 AssertReturn(VALID_PTR (deviceNode), VERR_INVALID_POINTER);
194 LogFlowFunc (("deviceNode=%s, isDVD=%d\n", deviceNode, isDVD));
195 struct stat statInfo;
196 bool retValue = false;
197
198 // sanity check
199 if (!deviceNode)
200 {
201 return false;
202 }
203
204 // first a simple stat() call
205 if (stat(deviceNode, &statInfo) < 0)
206 {
207 return false;
208 } else
209 {
210 if (isDVD)
211 {
212 if (S_ISCHR(statInfo.st_mode) || S_ISBLK(statInfo.st_mode))
213 {
214 int fileHandle;
215 // now try to open the device
216 fileHandle = open(deviceNode, O_RDONLY | O_NONBLOCK, 0);
217 if (fileHandle >= 0)
218 {
219 cdrom_subchnl cdChannelInfo;
220 cdChannelInfo.cdsc_format = CDROM_MSF;
221 // this call will finally reveal the whole truth
222#ifdef RT_OS_LINUX
223 if ((ioctl(fileHandle, CDROMSUBCHNL, &cdChannelInfo) == 0) ||
224 (errno == EIO) || (errno == ENOENT) ||
225 (errno == EINVAL) || (errno == ENOMEDIUM))
226#endif
227 {
228 retValue = true;
229 }
230 close(fileHandle);
231 }
232 }
233 } else
234 {
235 // floppy case
236 if (S_ISCHR(statInfo.st_mode) || S_ISBLK(statInfo.st_mode))
237 {
238 /// @todo do some more testing, maybe a nice IOCTL!
239 retValue = true;
240 }
241 }
242 }
243 LogFlowFunc (("retValue=%d\n", retValue));
244 return retValue;
245}
246#else /* !RT_OS_LINUX */
247# error Port me! Copying code over from HostImpl.cpp should be most of the job though.
248#endif /* !RT_OS_LINUX */
249
250/**
251 * Extract the names of drives from an environment variable and add them to a
252 * list if they are valid.
253 * @returns iprt status code
254 * @param pszVar the name of the environment variable. The variable
255 * value should be a list of device node names, separated
256 * by ':' characters.
257 * @param pList the list to append the drives found to
258 * @param isDVD are we looking for DVD drives or for floppies?
259 * @param pfSuccess this will be set to true if we found at least one drive
260 * and to false otherwise. Optional.
261 */
262/* static */
263int getDriveInfoFromEnv(const char *pszVar, DriveInfoList *pList,
264 bool isDVD, bool *pfSuccess)
265{
266 AssertReturn( VALID_PTR (pszVar) && VALID_PTR (pList)
267 && (pfSuccess == NULL || VALID_PTR (pfSuccess)),
268 VERR_INVALID_POINTER);
269 LogFlowFunc (("pszVar=%s, pList=%p, isDVD=%d, pfSuccess=%p\n", pszVar,
270 pList, isDVD, pfSuccess));
271 int rc = VINF_SUCCESS;
272 bool success = false;
273 RTMemAutoPtr<char, RTStrFree> drive;
274 const char *pszValue = RTEnvGet (pszVar);
275 if (pszValue != NULL)
276 {
277 drive = RTStrDup (pszValue);
278 if (!drive)
279 rc = VERR_NO_MEMORY;
280 }
281 if (pszValue != NULL && RT_SUCCESS (rc))
282 {
283 char *pDrive = drive.get();
284 char *pDriveNext = strchr (pDrive, ':');
285 while (pDrive != NULL && *pDrive != '\0')
286 {
287 if (pDriveNext != NULL)
288 *pDriveNext = '\0';
289 if (validateDevice(pDrive, isDVD))
290 {
291 pList->push_back (DriveInfo (pDrive));
292 success = true;
293 }
294 if (pDriveNext != NULL)
295 {
296 pDrive = pDriveNext + 1;
297 pDriveNext = strchr (pDrive, ':');
298 }
299 else
300 pDrive = NULL;
301 }
302 }
303 if (pfSuccess != NULL)
304 *pfSuccess = success;
305 LogFlowFunc (("rc=%Rrc, success=%d\n", rc, success));
306 return rc;
307}
308
309#ifdef RT_OS_LINUX
310/**
311 * Helper function to parse the given mount file and add found entries
312 */
313/* static */
314int getDVDInfoFromMTab(char *mountTable, DriveInfoList *pList)
315{
316 AssertReturn(VALID_PTR (mountTable) && VALID_PTR (pList),
317 VERR_INVALID_POINTER);
318#ifdef RT_OS_LINUX
319 LogFlowFunc (("mountTable=%s, pList=%p\n", mountTable, pList));
320 int rc = VINF_SUCCESS;
321 FILE *mtab = setmntent(mountTable, "r");
322 if (mtab)
323 {
324 struct mntent *mntent;
325 RTMemAutoPtr <char, RTStrFree> mnt_type, mnt_dev;
326 char *tmp;
327 while (RT_SUCCESS (rc) && (mntent = getmntent(mtab)))
328 {
329 mnt_type = RTStrDup (mntent->mnt_type);
330 mnt_dev = RTStrDup (mntent->mnt_fsname);
331 if (!mnt_type || !mnt_dev)
332 rc = VERR_NO_MEMORY;
333 // supermount fs case
334 if (RT_SUCCESS (rc) && strcmp(mnt_type.get(), "supermount") == 0)
335 {
336 tmp = strstr(mntent->mnt_opts, "fs=");
337 if (tmp)
338 {
339 mnt_type = RTStrDup(tmp + strlen("fs="));
340 if (!mnt_type)
341 rc = VERR_NO_MEMORY;
342 else
343 {
344 tmp = strchr(mnt_type.get(), ',');
345 if (tmp)
346 *tmp = '\0';
347 }
348 }
349 tmp = strstr(mntent->mnt_opts, "dev=");
350 if (tmp)
351 {
352 mnt_dev = RTStrDup(tmp + strlen("dev="));
353 if (!mnt_dev)
354 rc = VERR_NO_MEMORY;
355 else
356 {
357 tmp = strchr(mnt_dev.get(), ',');
358 if (tmp)
359 *tmp = '\0';
360 }
361 }
362 }
363 // use strstr here to cover things fs types like "udf,iso9660"
364 if (RT_SUCCESS (rc) && strstr(mnt_type.get(), "iso9660") == 0)
365 {
366 if (validateDevice(mnt_dev.get(), true))
367 {
368 bool insert = true;
369 struct stat srcInfo;
370 if (stat (mnt_dev.get(), &srcInfo) < 0)
371 insert = false;
372 for (DriveInfoList::const_iterator it = pList->begin();
373 insert && it != pList->end(); ++it)
374 {
375 struct stat destInfo;
376 if ( (stat (it->mDevice.c_str(), &destInfo) == 0)
377 && (srcInfo.st_rdev == destInfo.st_rdev))
378 insert = false;
379 }
380 if (insert)
381 pList->push_back (DriveInfo (mnt_dev.get()));
382 }
383 }
384 }
385 endmntent(mtab);
386 }
387 return rc;
388#endif
389}
390
391#endif /* RT_OS_LINUX */
392
393#if defined(RT_OS_LINUX) && defined(VBOX_WITH_DBUS)
394/* Linux, load libdbus statically */
395
396/** Wrapper class around DBusError for automatic cleanup */
397class autoDBusError
398{
399 DBusError mError;
400public:
401 autoDBusError () { dbus_error_init (&mError); }
402 ~autoDBusError ()
403 {
404 if (IsSet())
405 dbus_error_free (&mError);
406 }
407 DBusError &get () { return mError; }
408 bool IsSet ()
409 {
410 Assert ((mError.name == NULL) == (mError.message == NULL));
411 return (mError.name != NULL);
412 }
413 bool HasName (const char *pszName)
414 {
415 Assert ((mError.name == NULL) == (mError.message == NULL));
416 return (RTStrCmp (mError.name, pszName) == 0);
417 }
418 void FlowLog ()
419 {
420 if (IsSet ())
421 LogFlow(("DBus error %s: %s\n", mError.name, mError.message));
422 }
423};
424
425/**
426 * Helper function for setting up a connection to hal
427 * @returns iprt status code
428 * @param ppConnection where to store the connection handle
429 */
430/* static */
431int halInit (DBusConnection **ppConnection)
432{
433 AssertReturn(VALID_PTR (ppConnection), VERR_INVALID_POINTER);
434 LogFlowFunc (("ppConnection=%p\n", ppConnection));
435 int rc = VINF_SUCCESS;
436 bool halSuccess = true;
437 autoDBusError dbusError;
438
439 RTMemAutoPtr <DBusConnection, VBoxDBusConnectionUnref> dbusConnection;
440 dbusConnection = dbus_bus_get (DBUS_BUS_SYSTEM, &dbusError.get());
441 if (!dbusConnection)
442 halSuccess = false;
443 if (halSuccess)
444 {
445 dbus_connection_set_exit_on_disconnect (dbusConnection.get(), false);
446 halSuccess = dbus_bus_name_has_owner (dbusConnection.get(),
447 "org.freedesktop.Hal", &dbusError.get());
448 }
449 if (halSuccess)
450 {
451 dbus_bus_add_match (dbusConnection.get(),
452 "type='signal',"
453 "interface='org.freedesktop.Hal.Manager',"
454 "sender='org.freedesktop.Hal',"
455 "path='/org/freedesktop/Hal/Manager'",
456 &dbusError.get());
457 halSuccess = !dbusError.IsSet();
458 }
459 if (dbusError.HasName (DBUS_ERROR_NO_MEMORY))
460 rc = VERR_NO_MEMORY;
461 *ppConnection = halSuccess ? dbusConnection.release() : NULL;
462 LogFlowFunc(("rc=%Rrc, *ppConnection=%p\n", rc, *ppConnection));
463 dbusError.FlowLog();
464 return rc;
465}
466
467/**
468 * Helper function for shutting down a connection to hal
469 * @param pConnection the connection handle
470 */
471/* static */
472void halShutdown (DBusConnection *pConnection)
473{
474 AssertReturnVoid(VALID_PTR (pConnection));
475 LogFlowFunc (("pConnection=%p\n", pConnection));
476 autoDBusError dbusError;
477
478 dbus_bus_remove_match (pConnection,
479 "type='signal',"
480 "interface='org.freedesktop.Hal.Manager',"
481 "sender='org.freedesktop.Hal',"
482 "path='/org/freedesktop/Hal/Manager'",
483 &dbusError.get());
484 dbus_connection_unref (pConnection);
485 LogFlowFunc(("returning\n"));
486 dbusError.FlowLog();
487}
488
489/**
490 * Find the UDIs of hal entries that contain Key=Value property.
491 * @returns iprt status code
492 * @param pConnection an initialised connection DBus
493 * @param pszKey the property key
494 * @param pszValue the property value
495 * @param ppMessage where to store the return DBus message. This must be
496 * parsed to get at the UDIs. NOT optional. The caller
497 * is responsible for freeing this.
498 */
499/* static */
500int halFindDeviceStringMatch (DBusConnection *pConnection, const char *pszKey,
501 const char *pszValue, DBusMessage **ppMessage)
502{
503 AssertReturn( VALID_PTR (pConnection) && VALID_PTR (pszKey)
504 && VALID_PTR (pszValue) && VALID_PTR (ppMessage),
505 VERR_INVALID_POINTER);
506 LogFlowFunc (("pConnection=%p, pszKey=%s, pszValue=%s, ppMessage=%p\n",
507 pConnection, pszKey, pszValue, ppMessage));
508 int rc = VINF_SUCCESS;
509 bool halSuccess = true;
510 autoDBusError dbusError;
511 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> message;
512 DBusMessage *pReply = NULL;
513 if (halSuccess && RT_SUCCESS (rc))
514 {
515 message = dbus_message_new_method_call ("org.freedesktop.Hal",
516 "/org/freedesktop/Hal/Manager",
517 "org.freedesktop.Hal.Manager",
518 "FindDeviceStringMatch");
519 if (!message)
520 rc = VERR_NO_MEMORY;
521 }
522 if (halSuccess && RT_SUCCESS (rc))
523 {
524 DBusMessageIter iterAppend;
525 dbus_message_iter_init_append (message.get(), &iterAppend);
526 dbus_message_iter_append_basic (&iterAppend, DBUS_TYPE_STRING, &pszKey);
527 dbus_message_iter_append_basic (&iterAppend, DBUS_TYPE_STRING, &pszValue);
528 pReply = dbus_connection_send_with_reply_and_block (pConnection,
529 message.get(), -1,
530 &dbusError.get());
531 if (pReply == NULL)
532 halSuccess = false;
533 }
534 *ppMessage = pReply;
535 LogFlowFunc (("rc=%Rrc, *ppMessage=%p\n", rc, *ppMessage));
536 dbusError.FlowLog();
537 return rc;
538}
539
540/**
541 * Read a set of string properties for a device. If some of the properties are
542 * not of type DBUS_TYPE_STRING then a NULL pointer will be returned for them.
543 * @returns iprt status code
544 * @param pConnection an initialised connection DBus
545 * @param pszUdi the Udi of the device
546 * @param cProps the number of property values to look up
547 * @param papszKeys the keys of the properties to be looked up
548 * @param papszValues where to store the values of the properties. The
549 * strings returned will be valid until the message
550 * returned in @a ppMessage is freed. Undefined if
551 * the message is NULL.
552 * @param ppMessage where to store the return DBus message. The caller
553 * is responsible for freeing this once they have
554 * finished with the value strings. NOT optional.
555 */
556/* static */
557int halGetPropertyStrings (DBusConnection *pConnection, const char *pszUdi,
558 size_t cProps, const char **papszKeys,
559 char **papszValues, DBusMessage **ppMessage)
560{
561 AssertReturn( VALID_PTR (pConnection) && VALID_PTR (pszUdi)
562 && VALID_PTR (papszKeys) && VALID_PTR (papszValues)
563 && VALID_PTR (ppMessage),
564 VERR_INVALID_POINTER);
565 LogFlowFunc (("pConnection=%p, pszUdi=%s, cProps=%llu, papszKeys=%p, papszValues=%p, ppMessage=%p\n",
566 pConnection, pszUdi, cProps, papszKeys, papszValues, ppMessage));
567 int rc = VINF_SUCCESS;
568 bool halSuccess = true;
569 autoDBusError dbusError;
570 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> message, reply;
571 DBusMessageIter iterGet, iterProps, iterKey, iterValue;
572
573 /* Initialise the return array to NULLs */
574 for (size_t i = 0; i < cProps; ++i)
575 papszValues[i] = NULL;
576 message = dbus_message_new_method_call ("org.freedesktop.Hal", pszUdi,
577 "org.freedesktop.Hal.Device",
578 "GetAllProperties");
579 if (!message)
580 rc = VERR_NO_MEMORY;
581 if (halSuccess && RT_SUCCESS (rc))
582 {
583 reply = dbus_connection_send_with_reply_and_block (pConnection,
584 message.get(), -1,
585 &dbusError.get());
586 if (!reply)
587 halSuccess = false;
588 }
589 if (halSuccess && RT_SUCCESS (rc))
590 {
591 dbus_message_iter_init (reply.get(), &iterGet);
592 if ( dbus_message_iter_get_arg_type (&iterGet) != DBUS_TYPE_ARRAY
593 && dbus_message_iter_get_element_type (&iterGet) != DBUS_TYPE_DICT_ENTRY)
594 halSuccess = false;
595 }
596 if (halSuccess && RT_SUCCESS (rc))
597 dbus_message_iter_recurse (&iterGet, &iterProps);
598 while ( halSuccess && RT_SUCCESS (rc)
599 && dbus_message_iter_get_arg_type (&iterProps)
600 == DBUS_TYPE_DICT_ENTRY)
601 {
602 const char *pszKey;
603 DBusMessageIter iterEntry, iterValue;
604 dbus_message_iter_recurse (&iterProps, &iterEntry);
605 dbus_message_iter_get_basic (&iterEntry, &pszKey);
606 dbus_message_iter_next (&iterEntry);
607 dbus_message_iter_recurse (&iterEntry, &iterValue);
608 for (size_t i = 0; i < cProps; ++i)
609 if (strcmp (pszKey, papszKeys[i]) == 0)
610 {
611 if (dbus_message_iter_get_arg_type (&iterValue) == DBUS_TYPE_STRING)
612 dbus_message_iter_get_basic (&iterValue, &papszValues[i]);
613 }
614 dbus_message_iter_next (&iterProps);
615 }
616 if (RT_SUCCESS (rc) && halSuccess)
617 *ppMessage = reply.release();
618 else
619 *ppMessage = NULL;
620 if (dbusError.HasName (DBUS_ERROR_NO_MEMORY))
621 rc = VERR_NO_MEMORY;
622 LogFlowFunc (("rc=%Rrc, *ppMessage=%p\n", rc, *ppMessage));
623 dbusError.FlowLog();
624 return rc;
625}
626
627/**
628 * Helper function to query the hal subsystem for information about drives
629 * attached to the system.
630 * @returns iprt status code
631 * @param pList where to add information about the drives detected
632 * @param isDVD are we looking for DVDs or floppies?
633 * @param pfSuccess will be set to true if all interactions with hal
634 * succeeded and to false otherwise. Optional.
635 *
636 * @returns IPRT status code
637 */
638/* static */
639int getDriveInfoFromHal(DriveInfoList *pList, bool isDVD, bool *pfSuccess)
640{
641 AssertReturn(VALID_PTR (pList) && (pfSuccess == NULL || VALID_PTR (pfSuccess)),
642 VERR_INVALID_POINTER);
643 LogFlowFunc (("pList=%p, isDVD=%d, pfSuccess=%p\n", pList, isDVD, pfSuccess));
644 DBusConnection *pConnection;
645 autoDBusError dbusError;
646 DBusMessage *pReply;
647 RTMemAutoPtr <DBusMessage, VBoxDBusMessageUnref> message, replyFind, replyGet;
648 DBusMessageIter iterFind, iterUdis;
649 bool halSuccess = true; /* Did something go wrong with hal or DBus? */
650 int rc = VINF_SUCCESS; /* Did a fatal error occur? */
651
652 rc = halInit (&pConnection);
653 RTMemAutoPtr <DBusConnection, halShutdown> dbusConnection;
654 dbusConnection = pConnection;
655 if (!dbusConnection)
656 halSuccess = false;
657 if (halSuccess && RT_SUCCESS (rc))
658 {
659 rc = halFindDeviceStringMatch (pConnection, "storage.drive_type",
660 isDVD ? "cdrom" : "floppy", &pReply);
661 replyFind = pReply;
662 if (!replyFind)
663 halSuccess = false;
664 }
665 if (halSuccess && RT_SUCCESS (rc))
666 {
667 dbus_message_iter_init (replyFind.get(), &iterFind);
668 if (dbus_message_iter_get_arg_type (&iterFind) != DBUS_TYPE_ARRAY)
669 halSuccess = false;
670 }
671 if (halSuccess && RT_SUCCESS (rc))
672 dbus_message_iter_recurse (&iterFind, &iterUdis);
673 for (; halSuccess && RT_SUCCESS (rc)
674 && dbus_message_iter_get_arg_type (&iterUdis) == DBUS_TYPE_STRING;
675 dbus_message_iter_next(&iterUdis))
676 {
677 /* Now get all properties from the iterator */
678 const char *pszUdi;
679 dbus_message_iter_get_basic (&iterUdis, &pszUdi);
680 static const char *papszKeys[] =
681 { "block.device", "info.product", "info.vendor" };
682 char *papszValues[RT_ELEMENTS (papszKeys)];
683 rc = halGetPropertyStrings (pConnection, pszUdi, RT_ELEMENTS (papszKeys),
684 papszKeys, papszValues, &pReply);
685 replyGet = pReply;
686 std::string description;
687 const char *pszDevice = papszValues[0], *pszProduct = papszValues[1],
688 *pszVendor = papszValues[2];
689 if (!!replyGet && pszDevice)
690 {
691 if ((pszVendor != NULL) && (pszVendor[0] != '\0'))
692 (description += pszVendor) += " ";
693 if ((pszProduct != NULL && pszProduct[0] != '\0'))
694 description += pszProduct;
695 pList->push_back (DriveInfo (pszDevice, pszUdi, description));
696 }
697 }
698 if (dbusError.HasName (DBUS_ERROR_NO_MEMORY))
699 rc = VERR_NO_MEMORY;
700 if (pfSuccess != NULL)
701 *pfSuccess = halSuccess;
702 LogFlow (("rc=%Rrc, halSuccess=%d\n", rc, halSuccess));
703 dbusError.FlowLog();
704 return rc;
705}
706#endif /* RT_OS_LINUX && VBOX_WITH_DBUS */
707
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