mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2024-12-26 15:36:14 +01:00
3ca9fd4dbb
2007-12-11 David Zeuthen <davidz@redhat.com> Rework how volumes, drives and volume monitoring is done. Previosly the model was GDrive <1-1> GVolume where a GDrive instance represented a mount point and a GVolume instance represented a mounted file system. This patch changes it the model to GDrive <1-N> GVolume <1-1> GMount where GMount now serves the purpose of the old GVolume and the new GVolume serves the purpose of the old GDrive. In addition the new GDrive interface is used to represent a collection of GVolume instances (typically partitions) and also contains utility to query the state of the physical drive the GDrive object represents (such as checking for media, polling the drive, ejecting the media etc.). Also implement mounting and unmounting in the Unix volume monitor backend. A subquent patch will introduce GDrive support for ejection of media. * Makefile.am: * gdrive.c: (g_drive_is_media_check_automatic), (g_drive_is_media_removable), (g_drive_has_media), (g_drive_can_poll_for_media), (g_drive_eject), (g_drive_eject_finish), (g_drive_poll_for_media), (g_drive_poll_for_media_finish): * gdrive.h: * gfile.c: (g_file_find_enclosing_mount): * gfile.h: * gio.symbols: * glocaldirectorymonitor.c: (g_local_directory_monitor_constructor), (mounts_changed): * glocalfile.c: (get_mount_info), (g_local_file_find_enclosing_mount), (g_local_file_file_iface_init): * gnativevolumemonitor.h: * gunionvolumemonitor.c: (get_mounts), (get_volumes), (get_connected_drives), (g_union_volume_monitor_class_init), (child_volume_added), (child_volume_removed), (child_volume_changed), (child_mount_added), (child_mount_removed), (child_mount_pre_unmount), (child_mount_changed), (child_drive_changed), (g_union_volume_monitor_add_monitor), (g_union_volume_monitor_remove_monitor), (_g_mount_get_for_mount_path): * gunixmounts.c: (g_unix_is_mount_path_system_internal), (guess_system_internal), (_g_get_unix_mounts), (_g_get_unix_mount_points), (g_get_unix_mount_at), (g_unix_mount_free), (g_unix_mount_compare), (g_unix_mount_get_mount_path), (g_unix_mount_get_device_path), (g_unix_mount_get_fs_type), (g_unix_mount_is_readonly), (g_unix_mount_is_system_internal), (g_unix_mount_guess_type), (type_to_icon), (g_unix_mount_guess_name), (g_unix_mount_guess_icon), (g_unix_mount_point_guess_name), (g_unix_mount_point_guess_icon), (_canonicalize_filename), (_resolve_symlink), (_resolve_dev_root): * gunixmounts.h: * gunixvolume.c: (g_unix_volume_finalize), (_g_unix_volume_new), (_g_unix_volume_disconnected), (_g_unix_volume_set_mount), (_g_unix_volume_unset_mount), (g_unix_volume_get_icon), (g_unix_volume_get_name), (g_unix_volume_can_mount), (g_unix_volume_get_drive), (g_unix_volume_get_mount), (_g_unix_volume_has_mount_path), (mount_cb), (mount_read_error), (g_unix_volume_mount), (g_unix_volume_mount_finish), (g_unix_volume_volume_iface_init): * gunixvolume.h: * gunixvolumemonitor.c: (g_unix_volume_monitor_finalize), (get_mounts), (get_volumes), (get_connected_drives), (get_mount_for_mount_path), (g_unix_volume_monitor_class_init), (mountpoints_changed), (mounts_changed), (g_unix_volume_monitor_init), (_g_unix_volume_monitor_lookup_volume_for_mount_path), (find_mount_by_mountpath), (update_volumes), (update_mounts): * gunixvolumemonitor.h: * gvolume.c: (g_volume_get_mount), (g_volume_can_mount), (g_volume_mount), (g_volume_mount_finish): * gvolume.h: * gvolumemonitor.c: (g_volume_monitor_class_init), (g_volume_monitor_get_connected_drives), (g_volume_monitor_get_volumes), (g_volume_monitor_get_mounts): * gvolumemonitor.h: svn path=/trunk/; revision=6095
442 lines
10 KiB
C
442 lines
10 KiB
C
/* GIO - GLib Input, Output and Streaming Library
|
|
*
|
|
* Copyright (C) 2006-2007 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General
|
|
* Public License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place, Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*
|
|
* Author: Alexander Larsson <alexl@redhat.com>
|
|
* David Zeuthen <davidz@redhat.com>
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include "gdrive.h"
|
|
#include "gsimpleasyncresult.h"
|
|
#include "glibintl.h"
|
|
|
|
#include "gioalias.h"
|
|
|
|
/**
|
|
* SECTION:gdrive
|
|
* @short_description: Virtual File System drive management
|
|
* @include: gio/gdrive.h
|
|
*
|
|
* #GDrive is a container class for #GVolume objects that stem from
|
|
* the same piece of media. As such, #GDrive abstracts a drive with
|
|
* (or without) removable media and provides operations for querying
|
|
* whether media is available, determing whether media change is
|
|
* automatically detected and ejecting the media.
|
|
*
|
|
* If the #GDrive reports that media isn't automatically detected, one
|
|
* can poll for media; typically one should not do this periodically
|
|
* as a poll for media operation is potententially expensive and may
|
|
* spin up the drive creating noise.
|
|
*
|
|
* For porting from GnomeVFS note that there is no equivalent of
|
|
* #GDrive in that API.
|
|
**/
|
|
|
|
static void g_drive_base_init (gpointer g_class);
|
|
static void g_drive_class_init (gpointer g_class,
|
|
gpointer class_data);
|
|
|
|
GType
|
|
g_drive_get_type (void)
|
|
{
|
|
static GType drive_type = 0;
|
|
|
|
if (! drive_type)
|
|
{
|
|
static const GTypeInfo drive_info =
|
|
{
|
|
sizeof (GDriveIface), /* class_size */
|
|
g_drive_base_init, /* base_init */
|
|
NULL, /* base_finalize */
|
|
g_drive_class_init,
|
|
NULL, /* class_finalize */
|
|
NULL, /* class_data */
|
|
0,
|
|
0, /* n_preallocs */
|
|
NULL
|
|
};
|
|
|
|
drive_type =
|
|
g_type_register_static (G_TYPE_INTERFACE, I_("GDrive"),
|
|
&drive_info, 0);
|
|
|
|
g_type_interface_add_prerequisite (drive_type, G_TYPE_OBJECT);
|
|
}
|
|
|
|
return drive_type;
|
|
}
|
|
|
|
static void
|
|
g_drive_class_init (gpointer g_class,
|
|
gpointer class_data)
|
|
{
|
|
}
|
|
|
|
static void
|
|
g_drive_base_init (gpointer g_class)
|
|
{
|
|
static gboolean initialized = FALSE;
|
|
|
|
if (! initialized)
|
|
{
|
|
/**
|
|
* GDrive::changed:
|
|
* @volume: a #GVolume.
|
|
*
|
|
* Emitted when the drive's state has changed.
|
|
*
|
|
**/
|
|
g_signal_new (I_("changed"),
|
|
G_TYPE_DRIVE,
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (GDriveIface, changed),
|
|
NULL, NULL,
|
|
g_cclosure_marshal_VOID__VOID,
|
|
G_TYPE_NONE, 0);
|
|
|
|
initialized = TRUE;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* g_drive_get_name:
|
|
* @drive: a #GDrive.
|
|
*
|
|
* Gets the name of @drive.
|
|
*
|
|
* Returns: a string containing @drive's name. The returned
|
|
* string should be freed when no longer needed.
|
|
**/
|
|
char *
|
|
g_drive_get_name (GDrive *drive)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), NULL);
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
return (* iface->get_name) (drive);
|
|
}
|
|
|
|
/**
|
|
* g_drive_get_icon:
|
|
* @drive: a #GDrive.
|
|
*
|
|
* Gets the icon for @drive.
|
|
*
|
|
* Returns: #GIcon for the @drive.
|
|
**/
|
|
GIcon *
|
|
g_drive_get_icon (GDrive *drive)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), NULL);
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
return (* iface->get_icon) (drive);
|
|
}
|
|
|
|
/**
|
|
* g_drive_has_volumes:
|
|
* @drive: a #GDrive.
|
|
*
|
|
* Check if @drive has any mountable volumes.
|
|
*
|
|
* Returns: %TRUE if the @drive contains volumes, %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
g_drive_has_volumes (GDrive *drive)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), FALSE);
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
return (* iface->has_volumes) (drive);
|
|
}
|
|
|
|
/**
|
|
* g_drive_get_volumes:
|
|
* @drive: a #GDrive.
|
|
*
|
|
* Get a list of mountable volumes for @drive.
|
|
*
|
|
* Returns: #GList containing any #GVolume<!---->s on the given @drive.
|
|
**/
|
|
GList *
|
|
g_drive_get_volumes (GDrive *drive)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), NULL);
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
return (* iface->get_volumes) (drive);
|
|
}
|
|
|
|
/**
|
|
* g_drive_is_media_check_automatic:
|
|
* @drive: a #GDrive.
|
|
*
|
|
* Checks if @drive is capabable of automatically detecting media changes.
|
|
*
|
|
* Returns: %TRUE if the @drive is capabable of automatically detecting media changes, %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
g_drive_is_media_check_automatic (GDrive *drive)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), FALSE);
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
return (* iface->is_media_check_automatic) (drive);
|
|
}
|
|
|
|
/**
|
|
* g_drive_is_media_removable:
|
|
* @drive: a #GDrive.
|
|
*
|
|
* Checks if the @drive supports removable media.
|
|
*
|
|
* Returns: %TRUE if @drive supports removable media, %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
g_drive_is_media_removable (GDrive *drive)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), FALSE);
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
return (* iface->is_media_removable) (drive);
|
|
}
|
|
|
|
/**
|
|
* g_drive_has_media:
|
|
* @drive: a #GDrive.
|
|
*
|
|
* Checks if the @drive has media. Note that the OS may not be polling
|
|
* the drive for media changes; see g_drive_is_media_check_automatic()
|
|
* for more details.
|
|
*
|
|
* Returns: %TRUE if @drive has media, %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
g_drive_has_media (GDrive *drive)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), FALSE);
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
return (* iface->has_media) (drive);
|
|
}
|
|
|
|
/**
|
|
* g_drive_can_eject:
|
|
* @drive: pointer to a #GDrive.
|
|
*
|
|
* Checks if a drive can be ejected.
|
|
*
|
|
* Returns: %TRUE if the @drive can be ejected. %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
g_drive_can_eject (GDrive *drive)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), FALSE);
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
if (iface->can_eject == NULL)
|
|
return FALSE;
|
|
|
|
return (* iface->can_eject) (drive);
|
|
}
|
|
|
|
/**
|
|
* g_drive_can_poll_for_media:
|
|
* @drive: a #GDrive.
|
|
*
|
|
* Checks if a drive can be polled for media changes.
|
|
*
|
|
* Returns: %TRUE if the @drive can be polled for media changes. %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
g_drive_can_poll_for_media (GDrive *drive)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), FALSE);
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
if (iface->poll_for_media == NULL)
|
|
return FALSE;
|
|
|
|
return (* iface->can_poll_for_media) (drive);
|
|
}
|
|
|
|
/**
|
|
* g_drive_eject:
|
|
* @drive: a #GDrive.
|
|
* @cancellable: optional #GCancellable object, %NULL to ignore.
|
|
* @callback: a #GAsyncReadyCallback.
|
|
* @user_data: a #gpointer.
|
|
*
|
|
* Ejects a drive.
|
|
*
|
|
**/
|
|
void
|
|
g_drive_eject (GDrive *drive,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer user_data)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_if_fail (G_IS_DRIVE (drive));
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
if (iface->eject == NULL)
|
|
{
|
|
g_simple_async_report_error_in_idle (G_OBJECT (drive), callback, user_data,
|
|
G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED,
|
|
_("drive doesn't implement eject"));
|
|
|
|
return;
|
|
}
|
|
|
|
(* iface->eject) (drive, cancellable, callback, user_data);
|
|
}
|
|
|
|
/**
|
|
* g_drive_eject_finish
|
|
* @drive: a #GDrive.
|
|
* @result: a #GAsyncResult.
|
|
* @error: a #GError.
|
|
*
|
|
* Finishes ejecting a drive.
|
|
*
|
|
* Returns: %TRUE if the drive has been ejected successfully,
|
|
* %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
g_drive_eject_finish (GDrive *drive,
|
|
GAsyncResult *result,
|
|
GError **error)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), FALSE);
|
|
g_return_val_if_fail (G_IS_ASYNC_RESULT (result), FALSE);
|
|
|
|
if (G_IS_SIMPLE_ASYNC_RESULT (result))
|
|
{
|
|
GSimpleAsyncResult *simple = G_SIMPLE_ASYNC_RESULT (result);
|
|
if (g_simple_async_result_propagate_error (simple, error))
|
|
return FALSE;
|
|
}
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
return (* iface->eject_finish) (drive, result, error);
|
|
}
|
|
|
|
/**
|
|
* g_drive_poll_for_media:
|
|
* @drive: a #GDrive.
|
|
* @cancellable: optional #GCancellable object, %NULL to ignore.
|
|
* @callback: a #GAsyncReadyCallback.
|
|
* @user_data: a #gpointer.
|
|
*
|
|
* Polls @drive to see if media has been inserted or removed.
|
|
*
|
|
**/
|
|
void
|
|
g_drive_poll_for_media (GDrive *drive,
|
|
GCancellable *cancellable,
|
|
GAsyncReadyCallback callback,
|
|
gpointer user_data)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_if_fail (G_IS_DRIVE (drive));
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
if (iface->poll_for_media == NULL)
|
|
{
|
|
g_simple_async_report_error_in_idle (G_OBJECT (drive), callback, user_data,
|
|
G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED,
|
|
_("drive doesn't implement polling for media"));
|
|
|
|
return;
|
|
}
|
|
|
|
(* iface->poll_for_media) (drive, cancellable, callback, user_data);
|
|
}
|
|
|
|
/**
|
|
* g_drive_poll_for_media_finish
|
|
* @drive: a #GDrive.
|
|
* @result: a #GAsyncResult.
|
|
* @error: a #GError.
|
|
*
|
|
* Finishes poll_for_mediaing a drive.
|
|
*
|
|
* Returns: %TRUE if the drive has been poll_for_mediaed successfully,
|
|
* %FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
g_drive_poll_for_media_finish (GDrive *drive,
|
|
GAsyncResult *result,
|
|
GError **error)
|
|
{
|
|
GDriveIface *iface;
|
|
|
|
g_return_val_if_fail (G_IS_DRIVE (drive), FALSE);
|
|
g_return_val_if_fail (G_IS_ASYNC_RESULT (result), FALSE);
|
|
|
|
if (G_IS_SIMPLE_ASYNC_RESULT (result))
|
|
{
|
|
GSimpleAsyncResult *simple = G_SIMPLE_ASYNC_RESULT (result);
|
|
if (g_simple_async_result_propagate_error (simple, error))
|
|
return FALSE;
|
|
}
|
|
|
|
iface = G_DRIVE_GET_IFACE (drive);
|
|
|
|
return (* iface->poll_for_media_finish) (drive, result, error);
|
|
}
|
|
|
|
#define __G_DRIVE_C__
|
|
#include "gioaliasdef.c"
|