mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-13 15:56:23 +01:00
Merge branch 'wip/oholy/gunixmountmonitor-thread-safe' into 'master'
gunixmounts: Make GUnixMountMonitor thread-safe Closes #2030 See merge request GNOME/glib!1357
This commit is contained in:
commit
4a1dcdfb7c
@ -152,7 +152,11 @@ static GList *_g_get_unix_mounts (void);
|
|||||||
static GList *_g_get_unix_mount_points (void);
|
static GList *_g_get_unix_mount_points (void);
|
||||||
static gboolean proc_mounts_watch_is_running (void);
|
static gboolean proc_mounts_watch_is_running (void);
|
||||||
|
|
||||||
|
G_LOCK_DEFINE_STATIC (proc_mounts_source);
|
||||||
|
|
||||||
|
/* Protected by proc_mounts_source lock */
|
||||||
static guint64 mount_poller_time = 0;
|
static guint64 mount_poller_time = 0;
|
||||||
|
static GSource *proc_mounts_watch_source;
|
||||||
|
|
||||||
#ifdef HAVE_SYS_MNTTAB_H
|
#ifdef HAVE_SYS_MNTTAB_H
|
||||||
#define MNTOPT_RO "ro"
|
#define MNTOPT_RO "ro"
|
||||||
@ -1485,18 +1489,21 @@ get_mounts_timestamp (void)
|
|||||||
{
|
{
|
||||||
const char *monitor_file;
|
const char *monitor_file;
|
||||||
struct stat buf;
|
struct stat buf;
|
||||||
|
guint64 timestamp = 0;
|
||||||
|
|
||||||
|
G_LOCK (proc_mounts_source);
|
||||||
|
|
||||||
monitor_file = get_mtab_monitor_file ();
|
monitor_file = get_mtab_monitor_file ();
|
||||||
/* Don't return mtime for /proc/ files */
|
/* Don't return mtime for /proc/ files */
|
||||||
if (monitor_file && !g_str_has_prefix (monitor_file, "/proc/"))
|
if (monitor_file && !g_str_has_prefix (monitor_file, "/proc/"))
|
||||||
{
|
{
|
||||||
if (stat (monitor_file, &buf) == 0)
|
if (stat (monitor_file, &buf) == 0)
|
||||||
return (guint64)buf.st_mtime;
|
timestamp = buf.st_mtime;
|
||||||
}
|
}
|
||||||
else if (proc_mounts_watch_is_running ())
|
else if (proc_mounts_watch_is_running ())
|
||||||
{
|
{
|
||||||
/* it's being monitored by poll, so return mount_poller_time */
|
/* it's being monitored by poll, so return mount_poller_time */
|
||||||
return mount_poller_time;
|
timestamp = mount_poller_time;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -1505,9 +1512,12 @@ get_mounts_timestamp (void)
|
|||||||
* return TRUE so any application caches depending on it (like eg.
|
* return TRUE so any application caches depending on it (like eg.
|
||||||
* the one in GIO) get invalidated and don't hold possibly outdated
|
* the one in GIO) get invalidated and don't hold possibly outdated
|
||||||
* data - see Bug 787731 */
|
* data - see Bug 787731 */
|
||||||
return (guint64) g_get_monotonic_time ();
|
timestamp = g_get_monotonic_time ();
|
||||||
}
|
}
|
||||||
return 0;
|
|
||||||
|
G_UNLOCK (proc_mounts_source);
|
||||||
|
|
||||||
|
return timestamp;
|
||||||
}
|
}
|
||||||
|
|
||||||
static guint64
|
static guint64
|
||||||
@ -1704,10 +1714,10 @@ G_DEFINE_TYPE (GUnixMountMonitor, g_unix_mount_monitor, G_TYPE_OBJECT)
|
|||||||
static GContextSpecificGroup mount_monitor_group;
|
static GContextSpecificGroup mount_monitor_group;
|
||||||
static GFileMonitor *fstab_monitor;
|
static GFileMonitor *fstab_monitor;
|
||||||
static GFileMonitor *mtab_monitor;
|
static GFileMonitor *mtab_monitor;
|
||||||
static GSource *proc_mounts_watch_source;
|
|
||||||
static GList *mount_poller_mounts;
|
static GList *mount_poller_mounts;
|
||||||
static guint mtab_file_changed_id;
|
static guint mtab_file_changed_id;
|
||||||
|
|
||||||
|
/* Called with proc_mounts_source lock held. */
|
||||||
static gboolean
|
static gboolean
|
||||||
proc_mounts_watch_is_running (void)
|
proc_mounts_watch_is_running (void)
|
||||||
{
|
{
|
||||||
@ -1746,6 +1756,9 @@ mtab_file_changed (GFileMonitor *monitor,
|
|||||||
GFileMonitorEvent event_type,
|
GFileMonitorEvent event_type,
|
||||||
gpointer user_data)
|
gpointer user_data)
|
||||||
{
|
{
|
||||||
|
GMainContext *context;
|
||||||
|
GSource *source;
|
||||||
|
|
||||||
if (event_type != G_FILE_MONITOR_EVENT_CHANGED &&
|
if (event_type != G_FILE_MONITOR_EVENT_CHANGED &&
|
||||||
event_type != G_FILE_MONITOR_EVENT_CREATED &&
|
event_type != G_FILE_MONITOR_EVENT_CREATED &&
|
||||||
event_type != G_FILE_MONITOR_EVENT_DELETED)
|
event_type != G_FILE_MONITOR_EVENT_DELETED)
|
||||||
@ -1758,7 +1771,16 @@ mtab_file_changed (GFileMonitor *monitor,
|
|||||||
if (mtab_file_changed_id > 0)
|
if (mtab_file_changed_id > 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
mtab_file_changed_id = g_idle_add (mtab_file_changed_cb, NULL);
|
context = g_main_context_get_thread_default ();
|
||||||
|
if (!context)
|
||||||
|
context = g_main_context_default ();
|
||||||
|
|
||||||
|
source = g_idle_source_new ();
|
||||||
|
g_source_set_priority (source, G_PRIORITY_DEFAULT);
|
||||||
|
g_source_set_callback (source, mtab_file_changed_cb, NULL, NULL);
|
||||||
|
g_source_set_name (source, "[gio] mtab_file_changed_cb");
|
||||||
|
g_source_attach (source, context);
|
||||||
|
g_source_unref (source);
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
@ -1768,7 +1790,10 @@ proc_mounts_changed (GIOChannel *channel,
|
|||||||
{
|
{
|
||||||
if (cond & G_IO_ERR)
|
if (cond & G_IO_ERR)
|
||||||
{
|
{
|
||||||
|
G_LOCK (proc_mounts_source);
|
||||||
mount_poller_time = (guint64) g_get_monotonic_time ();
|
mount_poller_time = (guint64) g_get_monotonic_time ();
|
||||||
|
G_UNLOCK (proc_mounts_source);
|
||||||
|
|
||||||
g_context_specific_group_emit (&mount_monitor_group, signals[MOUNTS_CHANGED]);
|
g_context_specific_group_emit (&mount_monitor_group, signals[MOUNTS_CHANGED]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1802,7 +1827,10 @@ mount_change_poller (gpointer user_data)
|
|||||||
|
|
||||||
if (has_changed)
|
if (has_changed)
|
||||||
{
|
{
|
||||||
|
G_LOCK (proc_mounts_source);
|
||||||
mount_poller_time = (guint64) g_get_monotonic_time ();
|
mount_poller_time = (guint64) g_get_monotonic_time ();
|
||||||
|
G_UNLOCK (proc_mounts_source);
|
||||||
|
|
||||||
g_context_specific_group_emit (&mount_monitor_group, signals[MOUNTPOINTS_CHANGED]);
|
g_context_specific_group_emit (&mount_monitor_group, signals[MOUNTPOINTS_CHANGED]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1819,11 +1847,13 @@ mount_monitor_stop (void)
|
|||||||
g_object_unref (fstab_monitor);
|
g_object_unref (fstab_monitor);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
G_LOCK (proc_mounts_source);
|
||||||
if (proc_mounts_watch_source != NULL)
|
if (proc_mounts_watch_source != NULL)
|
||||||
{
|
{
|
||||||
g_source_destroy (proc_mounts_watch_source);
|
g_source_destroy (proc_mounts_watch_source);
|
||||||
proc_mounts_watch_source = NULL;
|
proc_mounts_watch_source = NULL;
|
||||||
}
|
}
|
||||||
|
G_UNLOCK (proc_mounts_source);
|
||||||
|
|
||||||
if (mtab_monitor)
|
if (mtab_monitor)
|
||||||
{
|
{
|
||||||
@ -1831,6 +1861,12 @@ mount_monitor_stop (void)
|
|||||||
g_object_unref (mtab_monitor);
|
g_object_unref (mtab_monitor);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (mtab_file_changed_id)
|
||||||
|
{
|
||||||
|
g_source_remove (mtab_file_changed_id);
|
||||||
|
mtab_file_changed_id = 0;
|
||||||
|
}
|
||||||
|
|
||||||
g_list_free_full (mount_poller_mounts, (GDestroyNotify) g_unix_mount_free);
|
g_list_free_full (mount_poller_mounts, (GDestroyNotify) g_unix_mount_free);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1869,7 +1905,10 @@ mount_monitor_start (void)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
G_LOCK (proc_mounts_source);
|
||||||
|
|
||||||
proc_mounts_watch_source = g_io_create_watch (proc_mounts_channel, G_IO_ERR);
|
proc_mounts_watch_source = g_io_create_watch (proc_mounts_channel, G_IO_ERR);
|
||||||
|
mount_poller_time = (guint64) g_get_monotonic_time ();
|
||||||
g_source_set_callback (proc_mounts_watch_source,
|
g_source_set_callback (proc_mounts_watch_source,
|
||||||
(GSourceFunc) proc_mounts_changed,
|
(GSourceFunc) proc_mounts_changed,
|
||||||
NULL, NULL);
|
NULL, NULL);
|
||||||
@ -1877,6 +1916,8 @@ mount_monitor_start (void)
|
|||||||
g_main_context_get_thread_default ());
|
g_main_context_get_thread_default ());
|
||||||
g_source_unref (proc_mounts_watch_source);
|
g_source_unref (proc_mounts_watch_source);
|
||||||
g_io_channel_unref (proc_mounts_channel);
|
g_io_channel_unref (proc_mounts_channel);
|
||||||
|
|
||||||
|
G_UNLOCK (proc_mounts_source);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -1889,6 +1930,8 @@ mount_monitor_start (void)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
G_LOCK (proc_mounts_source);
|
||||||
|
|
||||||
proc_mounts_watch_source = g_timeout_source_new_seconds (3);
|
proc_mounts_watch_source = g_timeout_source_new_seconds (3);
|
||||||
mount_poller_mounts = _g_get_unix_mounts ();
|
mount_poller_mounts = _g_get_unix_mounts ();
|
||||||
mount_poller_time = (guint64)g_get_monotonic_time ();
|
mount_poller_time = (guint64)g_get_monotonic_time ();
|
||||||
@ -1898,6 +1941,8 @@ mount_monitor_start (void)
|
|||||||
g_source_attach (proc_mounts_watch_source,
|
g_source_attach (proc_mounts_watch_source,
|
||||||
g_main_context_get_thread_default ());
|
g_main_context_get_thread_default ());
|
||||||
g_source_unref (proc_mounts_watch_source);
|
g_source_unref (proc_mounts_watch_source);
|
||||||
|
|
||||||
|
G_UNLOCK (proc_mounts_source);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user