mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-12 15:36:17 +01:00
aa39a0557c
- Stop using a custom thread for listening to kqueue(2) events. Instead call kevent(2) in non blocking mode in a monitor callback. Under the hood poll(2) is used to figure out if new events are available. - Do not use a socketpair with a custom protocol requiring 2 supplementary context switches per event to commicate between multiple threads. Calling kevent(2), in non blocking mode, to add/remove events is fine from any context. - Add kqueue(2) events without the EV_ONESHOT flag. This removes a race where some notifications were lost because events had to be re-added for every new notification. - Get rid of the global hash table and its associated lock and races. Use the 'cookie' argument of kevent(2) to pass the associated descriptor when registering an event. - Fix _kh_file_appeared_cb() by properly passing a monitor instead of a source to g_file_monitor_emit_event(). - Properly refcount sources. - Remove a lot of abstraction making it harder to fix the remaining issues. https://bugzilla.gnome.org/show_bug.cgi?id=739424
172 lines
4.6 KiB
C
172 lines
4.6 KiB
C
/*******************************************************************************
|
|
Copyright (c) 2011, 2012 Dmitry Matveev <me@dmitrymatveev.co.uk>
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
THE SOFTWARE.
|
|
*******************************************************************************/
|
|
|
|
#include <glib.h>
|
|
|
|
#include "kqueue-helper.h"
|
|
|
|
|
|
#define SCAN_MISSING_TIME 4 /* 1/4 Hz */
|
|
|
|
void _kh_file_appeared_cb (kqueue_sub *sub);
|
|
|
|
static gboolean km_scan_missing (gpointer user_data);
|
|
|
|
static gboolean km_debug_enabled = FALSE;
|
|
#define KM_W if (km_debug_enabled) g_warning
|
|
|
|
static GSList *missing_subs_list = NULL;
|
|
G_LOCK_DEFINE_STATIC (missing_lock);
|
|
|
|
static volatile gboolean scan_missing_running = FALSE;
|
|
|
|
|
|
/**
|
|
* _km_add_missing:
|
|
* @sub: a #kqueue_sub
|
|
*
|
|
* Adds a subscription to the missing files list.
|
|
**/
|
|
void
|
|
_km_add_missing (kqueue_sub *sub)
|
|
{
|
|
G_LOCK (missing_lock);
|
|
if (g_slist_find (missing_subs_list, sub))
|
|
{
|
|
KM_W ("asked to add %s to missing list but it's already on the list!\n", sub->filename);
|
|
G_UNLOCK (missing_lock);
|
|
return;
|
|
}
|
|
|
|
KM_W ("adding %s to missing list\n", sub->filename);
|
|
missing_subs_list = g_slist_prepend (missing_subs_list, sub);
|
|
G_UNLOCK (missing_lock);
|
|
|
|
if (!scan_missing_running)
|
|
{
|
|
scan_missing_running = TRUE;
|
|
g_timeout_add_seconds (SCAN_MISSING_TIME, km_scan_missing, NULL);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* _kh_file_appeared_cb:
|
|
* @sub: a #kqueue_sub
|
|
*
|
|
* A callback function for kqueue-missing subsystem.
|
|
*
|
|
* Signals that a missing file has finally appeared in the filesystem.
|
|
* Emits %G_FILE_MONITOR_EVENT_CREATED.
|
|
**/
|
|
void
|
|
_kh_file_appeared_cb (kqueue_sub *sub)
|
|
{
|
|
GFile *child;
|
|
|
|
g_assert (sub != NULL);
|
|
g_assert (sub->filename);
|
|
|
|
if (!g_file_test (sub->filename, G_FILE_TEST_EXISTS))
|
|
return;
|
|
|
|
child = g_file_new_for_path (sub->filename);
|
|
|
|
g_file_monitor_emit_event (G_FILE_MONITOR (sub->mon),
|
|
child,
|
|
NULL,
|
|
G_FILE_MONITOR_EVENT_CREATED);
|
|
|
|
g_object_unref (child);
|
|
}
|
|
|
|
/**
|
|
* km_scan_missing:
|
|
* @user_data: unused
|
|
*
|
|
* The core missing files watching routine.
|
|
*
|
|
* Traverses through a list of missing files, tries to start watching each with
|
|
* kqueue, removes the appropriate entry and invokes a user callback if the file
|
|
* has appeared.
|
|
*
|
|
* Returns: %FALSE if no missing files left, %TRUE otherwise.
|
|
**/
|
|
static gboolean
|
|
km_scan_missing (gpointer user_data)
|
|
{
|
|
GSList *head;
|
|
GSList *not_missing = NULL;
|
|
gboolean retval = FALSE;
|
|
|
|
G_LOCK (missing_lock);
|
|
|
|
if (missing_subs_list)
|
|
KM_W ("we have a job");
|
|
|
|
for (head = missing_subs_list; head; head = head->next)
|
|
{
|
|
kqueue_sub *sub = (kqueue_sub *) head->data;
|
|
g_assert (sub != NULL);
|
|
g_assert (sub->filename != NULL);
|
|
|
|
if (_kqsub_start_watching (sub))
|
|
{
|
|
KM_W ("file %s now exists, starting watching", sub->filename);
|
|
_kh_file_appeared_cb (sub);
|
|
not_missing = g_slist_prepend (not_missing, head);
|
|
}
|
|
}
|
|
|
|
for (head = not_missing; head; head = head->next)
|
|
{
|
|
GSList *link = (GSList *) head->data;
|
|
missing_subs_list = g_slist_remove_link (missing_subs_list, link);
|
|
}
|
|
g_slist_free (not_missing);
|
|
|
|
if (missing_subs_list == NULL)
|
|
{
|
|
scan_missing_running = FALSE;
|
|
retval = FALSE;
|
|
}
|
|
else
|
|
retval = TRUE;
|
|
|
|
G_UNLOCK (missing_lock);
|
|
return retval;
|
|
}
|
|
|
|
|
|
/**
|
|
* _km_remove:
|
|
* @sub: a #kqueue_sub
|
|
*
|
|
* Removes a subscription from a list of missing files.
|
|
**/
|
|
void
|
|
_km_remove (kqueue_sub *sub)
|
|
{
|
|
G_LOCK (missing_lock);
|
|
missing_subs_list = g_slist_remove (missing_subs_list, sub);
|
|
G_UNLOCK (missing_lock);
|
|
}
|