summaryrefslogtreecommitdiff
path: root/gio/gunionvolumemonitor.c
diff options
context:
space:
mode:
authorDavid Zeuthen <davidz@redhat.com>2007-12-17 09:31:08 +0000
committerAlexander Larsson <alexl@src.gnome.org>2007-12-17 09:31:08 +0000
commitf4add8f0faf4a8c2ca1cdb4f99f5f3d0d66aaff8 (patch)
tree7b3dd83e827af09ff5750c0d90ac540cf9dd487c /gio/gunionvolumemonitor.c
parent96a0e238e5f8854ff0f4207e51bc0beed52d6c2f (diff)
downloadglib-f4add8f0faf4a8c2ca1cdb4f99f5f3d0d66aaff8.tar.gz
glib-f4add8f0faf4a8c2ca1cdb4f99f5f3d0d66aaff8.tar.bz2
glib-f4add8f0faf4a8c2ca1cdb4f99f5f3d0d66aaff8.zip
Provide eject() on both GMount and GVolume and utility functions to guess
2007-12-14 David Zeuthen <davidz@redhat.com> * Makefile.am: * gio.symbols: * gmount.c: (g_mount_get_uuid), (g_mount_can_eject), (g_mount_eject), (g_mount_eject_finish): * gmount.h: * gunionvolumemonitor.c: (g_union_volume_monitor_finalize), (get_volume_for_uuid), (get_mount_for_uuid), (g_union_volume_monitor_class_init), (get_default_native_type_with_exclude), (get_default_native_type), (get_native_type), (update_native_type), (g_union_volume_monitor_init), (_g_mount_get_for_mount_path): * gunixmount.c: (_g_unix_mount_new), (g_unix_mount_get_uuid), (g_unix_mount_can_eject), (eject_unmount_cb), (eject_unmount_read_error), (eject_unmount_do), (g_unix_mount_unmount), (g_unix_mount_eject), (g_unix_mount_eject_finish), (g_unix_mount_mount_iface_init): * gunixmounts.c: (g_unix_mount_guess_can_eject), (g_unix_mount_point_guess_can_eject): * gunixmounts.h: * gunixvolume.c: (_g_unix_volume_new), (g_unix_volume_get_uuid), (g_unix_volume_can_eject), (g_unix_volume_get_drive), (eject_mount_cb), (eject_mount_read_error), (eject_mount_do), (g_unix_volume_mount), (g_unix_volume_eject), (g_unix_volume_eject_finish), (g_unix_volume_volume_iface_init): * gunixvolumemonitor.c: (get_volume_for_uuid), (get_mount_for_uuid), (g_unix_volume_monitor_class_init), (update_mounts): * gvolume.c: (g_volume_get_uuid), (g_volume_can_eject), (g_volume_eject), (g_volume_eject_finish): * gvolume.h: * gvolumemonitor.c: (g_volume_monitor_get_volume_for_uuid), (g_volume_monitor_get_mount_for_uuid): * gvolumemonitor.h: Provide eject() on both GMount and GVolume and utility functions to guess whether a GUnixMountPoint or GUnixMountEntry should be ejected. Introduce the concept of UUID's and wire it into GVolume and GMount and provide API on GVolumeMonitor to find such instances. Also handle the case where an external GNativeVolumeMonitor fails to initialize. Lock around the _g_get_mount_for_mount_path() function such that volume monitor implementations won't have to do locking themselves. svn path=/trunk/; revision=6140
Diffstat (limited to 'gio/gunionvolumemonitor.c')
-rw-r--r--gio/gunionvolumemonitor.c156
1 files changed, 132 insertions, 24 deletions
diff --git a/gio/gunionvolumemonitor.c b/gio/gunionvolumemonitor.c
index 37e48cdf7..92ed3058e 100644
--- a/gio/gunionvolumemonitor.c
+++ b/gio/gunionvolumemonitor.c
@@ -59,12 +59,17 @@ static void
g_union_volume_monitor_finalize (GObject *object)
{
GUnionVolumeMonitor *monitor;
-
+ GVolumeMonitor *child_monitor;
+
monitor = G_UNION_VOLUME_MONITOR (object);
- while (monitor->monitors != NULL)
- g_union_volume_monitor_remove_monitor (monitor,
- monitor->monitors->data);
+ while (monitor->monitors != NULL) {
+ child_monitor = monitor->monitors->data;
+ g_union_volume_monitor_remove_monitor (monitor,
+ child_monitor);
+ g_object_unref (child_monitor);
+ }
+
if (G_OBJECT_CLASS (g_union_volume_monitor_parent_class)->finalize)
(*G_OBJECT_CLASS (g_union_volume_monitor_parent_class)->finalize) (object);
@@ -163,6 +168,64 @@ get_connected_drives (GVolumeMonitor *volume_monitor)
return res;
}
+static GVolume *
+get_volume_for_uuid (GVolumeMonitor *volume_monitor, const char *uuid)
+{
+ GUnionVolumeMonitor *monitor;
+ GVolumeMonitor *child_monitor;
+ GVolume *volume;
+ GList *l;
+
+ monitor = G_UNION_VOLUME_MONITOR (volume_monitor);
+
+ volume = NULL;
+
+ G_LOCK (the_volume_monitor);
+
+ for (l = monitor->monitors; l != NULL; l = l->next)
+ {
+ child_monitor = l->data;
+
+ volume = g_volume_monitor_get_volume_for_uuid (child_monitor, uuid);
+ if (volume != NULL)
+ break;
+
+ }
+
+ G_UNLOCK (the_volume_monitor);
+
+ return volume;
+}
+
+static GMount *
+get_mount_for_uuid (GVolumeMonitor *volume_monitor, const char *uuid)
+{
+ GUnionVolumeMonitor *monitor;
+ GVolumeMonitor *child_monitor;
+ GMount *mount;
+ GList *l;
+
+ monitor = G_UNION_VOLUME_MONITOR (volume_monitor);
+
+ mount = NULL;
+
+ G_LOCK (the_volume_monitor);
+
+ for (l = monitor->monitors; l != NULL; l = l->next)
+ {
+ child_monitor = l->data;
+
+ mount = g_volume_monitor_get_mount_for_uuid (child_monitor, uuid);
+ if (mount != NULL)
+ break;
+
+ }
+
+ G_UNLOCK (the_volume_monitor);
+
+ return mount;
+}
+
static void
g_union_volume_monitor_class_init (GUnionVolumeMonitorClass *klass)
{
@@ -175,6 +238,8 @@ g_union_volume_monitor_class_init (GUnionVolumeMonitorClass *klass)
monitor_class->get_connected_drives = get_connected_drives;
monitor_class->get_volumes = get_volumes;
monitor_class->get_mounts = get_mounts;
+ monitor_class->get_volume_for_uuid = get_volume_for_uuid;
+ monitor_class->get_mount_for_uuid = get_mount_for_uuid;
}
static void
@@ -325,14 +390,13 @@ g_union_volume_monitor_remove_monitor (GUnionVolumeMonitor *union_monitor,
g_signal_handlers_disconnect_by_func (child_monitor, child_drive_changed, union_monitor);
}
-static gpointer
-get_default_native_type (gpointer data)
+static GType
+get_default_native_type_with_exclude (GType type_to_exclude)
{
GNativeVolumeMonitorClass *klass;
GType *monitors;
guint n_monitors;
GType native_type;
- GType *ret = (GType *) data;
int native_prio;
int i;
@@ -347,39 +411,57 @@ get_default_native_type (gpointer data)
/* Ensure vfs in modules loaded */
_g_io_modules_ensure_loaded ();
-
+
monitors = g_type_children (G_TYPE_NATIVE_VOLUME_MONITOR, &n_monitors);
native_type = 0;
native_prio = -1;
for (i = 0; i < n_monitors; i++)
{
- klass = G_NATIVE_VOLUME_MONITOR_CLASS (g_type_class_ref (monitors[i]));
- if (klass->priority > native_prio)
+ if (monitors[i] != type_to_exclude)
{
- native_prio = klass->priority;
- native_type = monitors[i];
- }
+ klass = G_NATIVE_VOLUME_MONITOR_CLASS (g_type_class_ref (monitors[i]));
+
+ if (klass->priority > native_prio)
+ {
+ native_prio = klass->priority;
+ native_type = monitors[i];
+ }
- g_type_class_unref (klass);
+ g_type_class_unref (klass);
+ }
}
g_free (monitors);
- *ret = native_type;
+ return native_type;
+}
+static gpointer
+get_default_native_type (gpointer data)
+{
+ GType *ret = (GType *) data;
+
+ *ret = get_default_native_type_with_exclude (G_TYPE_INVALID);
return NULL;
}
+static GOnce _once_init = G_ONCE_INIT;
+static GType _type = G_TYPE_INVALID;
+
static GType
-get_native_type (void)
+get_native_type ()
{
- static GOnce once_init = G_ONCE_INIT;
- static GType type = G_TYPE_INVALID;
- g_once (&once_init, get_default_native_type, &type);
+ g_once (&_once_init, get_default_native_type, &_type);
- return type;
+ return _type;
+}
+
+static void
+update_native_type (GType type)
+{
+ _type = type;
}
static void
@@ -396,8 +478,22 @@ g_union_volume_monitor_init (GUnionVolumeMonitor *union_monitor)
if (native_type != G_TYPE_INVALID)
{
monitor = g_object_new (native_type, NULL);
- g_union_volume_monitor_add_monitor (union_monitor, monitor);
- g_object_unref (monitor);
+ /* A native file monitor (the hal one if hald isn't running for
+ * example) may very well fail so handle falling back to the
+ * native one shipped with gio (e.g. GUnixVolumeMonitor)
+ */
+ if (monitor == NULL)
+ {
+ native_type = get_default_native_type_with_exclude (native_type);
+ monitor = g_object_new (native_type, NULL);
+ }
+
+ if (monitor != NULL)
+ {
+ g_union_volume_monitor_add_monitor (union_monitor, monitor);
+ g_object_unref (monitor);
+ update_native_type (native_type);
+ }
}
monitors = g_type_children (G_TYPE_VOLUME_MONITOR, &n_monitors);
@@ -474,10 +570,22 @@ _g_mount_get_for_mount_path (const char *mount_path)
return NULL;
mount = NULL;
-
+
klass = G_NATIVE_VOLUME_MONITOR_CLASS (g_type_class_ref (native_type));
if (klass->get_mount_for_mount_path)
- mount = klass->get_mount_for_mount_path (mount_path);
+ {
+ G_LOCK (the_volume_monitor);
+ mount = klass->get_mount_for_mount_path (mount_path);
+ G_UNLOCK (the_volume_monitor);
+ }
+
+ /* TODO: How do we know this succeeded? Keep in mind that the native
+ * volume monitor may fail (e.g. not being able to connect to
+ * hald). Is the get_mount_for_mount_path() method allowed to
+ * return NULL? Seems like it is ... probably the method needs
+ * to take a boolean and write if it succeeds or not.. Messy.
+ * Very messy.
+ */
g_type_class_unref (klass);