diff options
author | David Zeuthen <davidz@redhat.com> | 2007-12-17 09:31:08 +0000 |
---|---|---|
committer | Alexander Larsson <alexl@src.gnome.org> | 2007-12-17 09:31:08 +0000 |
commit | f4add8f0faf4a8c2ca1cdb4f99f5f3d0d66aaff8 (patch) | |
tree | 7b3dd83e827af09ff5750c0d90ac540cf9dd487c /gio/gunionvolumemonitor.c | |
parent | 96a0e238e5f8854ff0f4207e51bc0beed52d6c2f (diff) | |
download | glib-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.c | 156 |
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); |