1098 lines
28 KiB
C++
1098 lines
28 KiB
C++
/*
|
|
* moofilewatch.h
|
|
*
|
|
* Copyright (C) 2004-2010 by Yevgen Muntyan <emuntyan@users.sourceforge.net>
|
|
*
|
|
* This file is part of medit. medit 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.1 of the License,
|
|
* or (at your option) any later version.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#define WANT_STAT_MONITOR
|
|
|
|
#ifdef __WIN32__
|
|
#include <windows.h>
|
|
#include <io.h>
|
|
#include <fcntl.h>
|
|
#endif
|
|
|
|
#ifndef __WIN32__
|
|
#include <time.h>
|
|
#endif
|
|
|
|
#include <mooglib/moo-glib.h>
|
|
#include <mooglib/moo-stat.h>
|
|
#include <sys/types.h>
|
|
#include "mooutils/mooutils-misc.h"
|
|
#include "mooutils/mooutils-mem.h"
|
|
#include "mooutils/moofilewatch.h"
|
|
#include "mooutils/mootype-macros.h"
|
|
#include "marshals.h"
|
|
#include "mooutils/mooutils-thread.h"
|
|
#include "mooutils/moolist.h"
|
|
|
|
#include <memory>
|
|
#include <moocpp/strutils.h>
|
|
|
|
using namespace moo;
|
|
|
|
#if 1
|
|
static void G_GNUC_PRINTF(1,2) DEBUG_PRINT (G_GNUC_UNUSED const char *format, ...)
|
|
{
|
|
}
|
|
#elif defined(__WIN32__)
|
|
#define DEBUG_PRINT _moo_message_async
|
|
#else
|
|
#define DEBUG_PRINT _moo_message
|
|
#endif
|
|
|
|
typedef struct {
|
|
guint id;
|
|
MooFileWatch *watch;
|
|
char *filename;
|
|
|
|
MooFileWatchCallback callback;
|
|
GDestroyNotify notify;
|
|
gpointer data;
|
|
|
|
MgwStatBuf statbuf;
|
|
|
|
guint isdir : 1;
|
|
guint alive : 1;
|
|
} Monitor;
|
|
|
|
struct WatchFuncs {
|
|
gboolean (*start) (MooFileWatch& watch,
|
|
GError **error);
|
|
gboolean (*shutdown) (MooFileWatch& watch,
|
|
GError **error);
|
|
|
|
gboolean (*start_monitor) (MooFileWatch& watch,
|
|
Monitor* monitor,
|
|
GError** error);
|
|
void (*stop_monitor) (MooFileWatch& watch,
|
|
Monitor& monitor);
|
|
};
|
|
|
|
MOO_DEFINE_SLIST(MonitorList, monitor_list, Monitor)
|
|
|
|
struct MooFileWatch::Impl
|
|
{
|
|
Impl();
|
|
|
|
guint id;
|
|
guint stat_timeout;
|
|
MonitorList *monitors;
|
|
GHashTable *requests; /* int -> Monitor* */
|
|
guint alive : 1;
|
|
};
|
|
|
|
typedef enum
|
|
{
|
|
MOO_FILE_WATCH_ERROR_CLOSED,
|
|
MOO_FILE_WATCH_ERROR_FAILED,
|
|
MOO_FILE_WATCH_ERROR_NOT_IMPLEMENTED,
|
|
MOO_FILE_WATCH_ERROR_TOO_MANY,
|
|
MOO_FILE_WATCH_ERROR_NOT_DIR,
|
|
MOO_FILE_WATCH_ERROR_IS_DIR,
|
|
MOO_FILE_WATCH_ERROR_NONEXISTENT,
|
|
MOO_FILE_WATCH_ERROR_BAD_FILENAME,
|
|
MOO_FILE_WATCH_ERROR_ACCESS_DENIED
|
|
} MooFileWatchError;
|
|
|
|
#define MOO_FILE_WATCH_ERROR (moo_file_watch_error_quark ())
|
|
MOO_DEFINE_QUARK_STATIC (moo-file-watch-error, moo_file_watch_error_quark)
|
|
|
|
#ifdef WANT_STAT_MONITOR
|
|
static gboolean watch_stat_start (MooFileWatch &watch,
|
|
GError** error);
|
|
static gboolean watch_stat_shutdown (MooFileWatch& watch,
|
|
GError** error);
|
|
static gboolean watch_stat_start_monitor (MooFileWatch& watch,
|
|
Monitor* monitor,
|
|
GError** error);
|
|
#endif
|
|
|
|
#ifdef __WIN32__
|
|
static gboolean watch_win32_start (MooFileWatch& watch,
|
|
GError** error);
|
|
static gboolean watch_win32_shutdown (MooFileWatch& watch,
|
|
GError** error);
|
|
static gboolean watch_win32_start_monitor (MooFileWatch& watch,
|
|
Monitor* monitor,
|
|
GError** error);
|
|
static void watch_win32_stop_monitor (MooFileWatch& watch,
|
|
Monitor& monitor);
|
|
#endif /* __WIN32__ */
|
|
|
|
static Monitor *monitor_new (MooFileWatch& watch,
|
|
const char* filename,
|
|
MooFileWatchCallback callback,
|
|
gpointer data,
|
|
GDestroyNotify notify);
|
|
static void monitor_free (Monitor* monitor);
|
|
|
|
|
|
static struct WatchFuncs watch_funcs = {
|
|
#if defined(__WIN32__)
|
|
watch_win32_start,
|
|
watch_win32_shutdown,
|
|
watch_win32_start_monitor,
|
|
watch_win32_stop_monitor
|
|
#else
|
|
watch_stat_start,
|
|
watch_stat_shutdown,
|
|
watch_stat_start_monitor,
|
|
NULL
|
|
#endif
|
|
};
|
|
|
|
|
|
static guint
|
|
get_new_monitor_id (void)
|
|
{
|
|
static guint id = 0;
|
|
|
|
if (!++id)
|
|
++id;
|
|
|
|
return id;
|
|
}
|
|
|
|
static guint
|
|
get_new_watch_id (void)
|
|
{
|
|
static guint id = 0;
|
|
|
|
if (!++id)
|
|
++id;
|
|
|
|
return id;
|
|
}
|
|
|
|
|
|
MooFileWatch::~MooFileWatch()
|
|
{
|
|
if (m_impl->alive)
|
|
{
|
|
g_warning ("finalizing open watch");
|
|
|
|
GError *error = NULL;
|
|
if (!close(&error))
|
|
{
|
|
g_warning ("error in moo_file_watch_close(): %s",
|
|
moo_error_message (error));
|
|
g_error_free (error);
|
|
}
|
|
}
|
|
|
|
g_hash_table_destroy(m_impl->requests);
|
|
}
|
|
|
|
|
|
MooFileWatch::Impl::Impl()
|
|
: id(get_new_watch_id())
|
|
, stat_timeout(0)
|
|
, monitors(nullptr)
|
|
, requests(nullptr)
|
|
, alive(false)
|
|
{
|
|
requests = g_hash_table_new (g_direct_hash, g_direct_equal);
|
|
}
|
|
|
|
MooFileWatch::MooFileWatch()
|
|
: m_ref(1)
|
|
, m_impl(make_unique<Impl>())
|
|
{
|
|
}
|
|
|
|
MooFileWatchPtr MooFileWatch::create(GError **error)
|
|
{
|
|
MooFileWatchPtr watch = MooFileWatchPtr::create();
|
|
|
|
if (!watch_funcs.start(*watch, error))
|
|
return nullptr;
|
|
|
|
watch->m_impl->alive = true;
|
|
return watch;
|
|
}
|
|
|
|
void MooFileWatch::ref()
|
|
{
|
|
++m_ref;
|
|
}
|
|
|
|
void MooFileWatch::unref()
|
|
{
|
|
if (--m_ref == 0)
|
|
delete this;
|
|
}
|
|
|
|
|
|
bool MooFileWatch::close(GError** error)
|
|
{
|
|
if (!m_impl->alive)
|
|
return true;
|
|
|
|
m_impl->alive = FALSE;
|
|
|
|
MonitorList *monitors = m_impl->monitors;
|
|
m_impl->monitors = NULL;
|
|
|
|
while (monitors)
|
|
{
|
|
Monitor *mon = monitors->data;
|
|
|
|
if (watch_funcs.stop_monitor)
|
|
watch_funcs.stop_monitor(*this, *mon);
|
|
|
|
monitor_free (mon);
|
|
monitors = monitor_list_delete_link (monitors, monitors);
|
|
}
|
|
|
|
return watch_funcs.shutdown(*this, error);
|
|
}
|
|
|
|
|
|
guint MooFileWatch::create_monitor(const char* filename,
|
|
MooFileWatchCallback callback,
|
|
gpointer data,
|
|
GDestroyNotify notify,
|
|
GError** error)
|
|
{
|
|
g_return_val_if_fail (filename != NULL, 0);
|
|
g_return_val_if_fail (callback != NULL, 0);
|
|
|
|
if (!m_impl->alive)
|
|
{
|
|
g_set_error (error, MOO_FILE_WATCH_ERROR,
|
|
MOO_FILE_WATCH_ERROR_CLOSED,
|
|
"MooFileWatch %u closed",
|
|
m_impl->id);
|
|
return 0;
|
|
}
|
|
|
|
Monitor *monitor = monitor_new(*this, filename, callback, data, notify);
|
|
|
|
if (!watch_funcs.start_monitor(*this, monitor, error))
|
|
{
|
|
monitor_free (monitor);
|
|
return 0;
|
|
}
|
|
|
|
monitor->alive = TRUE;
|
|
m_impl->monitors = monitor_list_prepend(m_impl->monitors, monitor);
|
|
g_hash_table_insert(m_impl->requests, GUINT_TO_POINTER(monitor->id), monitor);
|
|
|
|
DEBUG_PRINT ("created monitor %d for '%s'", monitor->id, monitor->filename);
|
|
|
|
return monitor->id;
|
|
}
|
|
|
|
|
|
void MooFileWatch::cancel_monitor(guint monitor_id)
|
|
{
|
|
Monitor *monitor = (Monitor*) g_hash_table_lookup(m_impl->requests,
|
|
GUINT_TO_POINTER (monitor_id));
|
|
g_return_if_fail (monitor != NULL);
|
|
|
|
m_impl->monitors = monitor_list_remove(m_impl->monitors, monitor);
|
|
g_hash_table_remove (m_impl->requests, GUINT_TO_POINTER (monitor->id));
|
|
|
|
if (monitor->alive)
|
|
DEBUG_PRINT ("stopping monitor %d for '%s'",
|
|
monitor->id, monitor->filename);
|
|
else
|
|
DEBUG_PRINT ("stopping dead monitor %d for '%s'",
|
|
monitor->id, monitor->filename);
|
|
|
|
if (monitor->alive && watch_funcs.stop_monitor)
|
|
watch_funcs.stop_monitor(*this, *monitor);
|
|
|
|
monitor_free(monitor);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_file_watch_emit_event (MooFileWatch *watch,
|
|
MooFileEvent *event,
|
|
Monitor *monitor)
|
|
{
|
|
watch->ref();
|
|
|
|
if (monitor->alive || event->code == MOO_FILE_EVENT_ERROR)
|
|
{
|
|
static const char *names[] = {
|
|
"changed", "created", "deleted", "error"
|
|
};
|
|
DEBUG_PRINT ("emitting event %s for %s", names[event->code], event->filename);
|
|
monitor->callback (*watch, event, monitor->data);
|
|
}
|
|
|
|
watch->unref();
|
|
}
|
|
|
|
|
|
static Monitor *
|
|
monitor_new (MooFileWatch& watch,
|
|
const char *filename,
|
|
MooFileWatchCallback callback,
|
|
gpointer data,
|
|
GDestroyNotify notify)
|
|
{
|
|
Monitor *mon;
|
|
|
|
mon = g_new0 (Monitor, 1);
|
|
mon->watch = &watch;
|
|
mon->filename = g_strdup (filename);
|
|
|
|
mon->callback = callback;
|
|
mon->notify = notify;
|
|
mon->data = data;
|
|
|
|
return mon;
|
|
}
|
|
|
|
|
|
static void
|
|
monitor_free (Monitor *monitor)
|
|
{
|
|
if (monitor)
|
|
{
|
|
if (monitor->notify)
|
|
monitor->notify (monitor->data);
|
|
g_free (monitor->filename);
|
|
g_free (monitor);
|
|
}
|
|
}
|
|
|
|
|
|
/*****************************************************************************/
|
|
/* stat()
|
|
*/
|
|
|
|
#ifdef WANT_STAT_MONITOR
|
|
|
|
#define MOO_STAT_PRIORITY G_PRIORITY_DEFAULT
|
|
#define MOO_STAT_TIMEOUT 500
|
|
|
|
static MooFileWatchError errno_to_file_error (mgw_errno_t code);
|
|
static gboolean do_stat (MooFileWatch* watch);
|
|
|
|
static gboolean
|
|
watch_stat_start(MooFileWatch& watch,
|
|
G_GNUC_UNUSED GError** error)
|
|
{
|
|
watch.ref();
|
|
watch.impl().stat_timeout =
|
|
gdk_threads_add_timeout_full (MOO_STAT_PRIORITY,
|
|
MOO_STAT_TIMEOUT,
|
|
(GSourceFunc) do_stat,
|
|
&watch, NULL);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
watch_stat_shutdown (MooFileWatch &watch,
|
|
G_GNUC_UNUSED GError **error)
|
|
{
|
|
if (watch.impl().stat_timeout)
|
|
{
|
|
g_source_remove(watch.impl().stat_timeout);
|
|
watch.impl().stat_timeout = 0;
|
|
watch.unref();
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
watch_stat_start_monitor (G_GNUC_UNUSED MooFileWatch& watch,
|
|
Monitor* monitor,
|
|
GError** error)
|
|
{
|
|
MgwStatBuf buf;
|
|
mgw_errno_t err;
|
|
|
|
g_return_val_if_fail (monitor->filename != NULL, FALSE);
|
|
|
|
if (mgw_stat (monitor->filename, &buf, &err) != 0)
|
|
{
|
|
g_set_error (error, MOO_FILE_WATCH_ERROR,
|
|
errno_to_file_error (err),
|
|
"stat: %s", mgw_strerror (err));
|
|
return FALSE;
|
|
}
|
|
|
|
monitor->isdir = buf.isdir;
|
|
|
|
#ifdef __WIN32__
|
|
if (monitor->isdir) /* it's fatal on windows */
|
|
{
|
|
char *display_name = g_filename_display_name (monitor->filename);
|
|
g_set_error (error, MOO_FILE_WATCH_ERROR,
|
|
MOO_FILE_WATCH_ERROR_FAILED,
|
|
"%s is a directory", display_name);
|
|
g_free (display_name);
|
|
return FALSE;
|
|
}
|
|
#endif
|
|
|
|
monitor->id = get_new_monitor_id ();
|
|
monitor->statbuf = buf;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
do_stat(MooFileWatch *watch)
|
|
{
|
|
MonitorList *lm;
|
|
GSList *list = NULL, *lid;
|
|
GSList *to_remove = NULL;
|
|
gboolean result = TRUE;
|
|
|
|
g_return_val_if_fail(watch != NULL, FALSE);
|
|
|
|
watch->ref();
|
|
|
|
if (!watch->impl().monitors)
|
|
goto out;
|
|
|
|
for (lm = watch->impl().monitors; lm != NULL; lm = lm->next)
|
|
{
|
|
Monitor *m = lm->data;
|
|
list = g_slist_prepend (list, GUINT_TO_POINTER (m->id));
|
|
}
|
|
|
|
/* Order of list is correct now, watch->monitors is last-added-first */
|
|
for (lid = list; lid != NULL; lid = lid->next)
|
|
{
|
|
gboolean do_emit = FALSE;
|
|
MooFileEvent event;
|
|
Monitor *monitor;
|
|
mgw_time_t old;
|
|
mgw_errno_t err;
|
|
|
|
monitor = (Monitor*) g_hash_table_lookup(watch->impl().requests, lid->data);
|
|
|
|
if (!monitor || !monitor->alive)
|
|
continue;
|
|
|
|
old = monitor->statbuf.mtime;
|
|
|
|
event.monitor_id = monitor->id;
|
|
event.filename = monitor->filename;
|
|
event.error = NULL;
|
|
|
|
if (mgw_stat (monitor->filename, &monitor->statbuf, &err) != 0)
|
|
{
|
|
if (err.value == MGW_ENOENT)
|
|
{
|
|
event.code = MOO_FILE_EVENT_DELETED;
|
|
to_remove = g_slist_prepend (to_remove, GUINT_TO_POINTER (monitor->id));
|
|
}
|
|
else
|
|
{
|
|
event.code = MOO_FILE_EVENT_ERROR;
|
|
g_set_error (&event.error, MOO_FILE_WATCH_ERROR,
|
|
errno_to_file_error (err),
|
|
"stat failed: %s",
|
|
mgw_strerror (err));
|
|
monitor->alive = FALSE;
|
|
}
|
|
|
|
do_emit = TRUE;
|
|
}
|
|
else if (monitor->statbuf.mtime.value > old.value)
|
|
{
|
|
event.code = MOO_FILE_EVENT_CHANGED;
|
|
do_emit = TRUE;
|
|
}
|
|
|
|
if (do_emit)
|
|
moo_file_watch_emit_event (watch, &event, monitor);
|
|
|
|
if (event.error)
|
|
g_error_free (event.error);
|
|
}
|
|
|
|
for (lid = to_remove; lid != NULL; lid = lid->next)
|
|
if (g_hash_table_lookup (watch->impl().requests, GUINT_TO_POINTER (lid->data)))
|
|
watch->cancel_monitor (GPOINTER_TO_UINT (lid->data));
|
|
|
|
g_slist_free (to_remove);
|
|
g_slist_free (list);
|
|
|
|
out:
|
|
watch->unref();
|
|
return result;
|
|
}
|
|
|
|
|
|
static MooFileWatchError
|
|
errno_to_file_error (mgw_errno_t code)
|
|
{
|
|
MooFileWatchError fcode = MOO_FILE_WATCH_ERROR_FAILED;
|
|
|
|
switch (code.value)
|
|
{
|
|
case MGW_EACCES:
|
|
fcode = MOO_FILE_WATCH_ERROR_ACCESS_DENIED;
|
|
break;
|
|
case MGW_ENAMETOOLONG:
|
|
fcode = MOO_FILE_WATCH_ERROR_BAD_FILENAME;
|
|
break;
|
|
case MGW_ENOENT:
|
|
fcode = MOO_FILE_WATCH_ERROR_NONEXISTENT;
|
|
break;
|
|
case MGW_ENOTDIR:
|
|
fcode = MOO_FILE_WATCH_ERROR_NOT_DIR;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return fcode;
|
|
}
|
|
|
|
|
|
#endif /* WANT_STAT_MONITOR */
|
|
|
|
/*****************************************************************************/
|
|
/* win32
|
|
*/
|
|
#ifdef __WIN32__
|
|
|
|
typedef struct {
|
|
guint watch_id;
|
|
guint request;
|
|
char *path;
|
|
} FAMThreadWatch;
|
|
|
|
typedef struct {
|
|
GMutex *lock;
|
|
GAsyncQueue *incoming;
|
|
HANDLE events[MAXIMUM_WAIT_OBJECTS];
|
|
FAMThreadWatch watches[MAXIMUM_WAIT_OBJECTS];
|
|
guint n_events;
|
|
gboolean running;
|
|
} FAMThread;
|
|
|
|
typedef enum {
|
|
COMMAND_ADD_PATH,
|
|
COMMAND_REMOVE_PATH
|
|
} FAMThreadCommandType;
|
|
|
|
typedef struct {
|
|
FAMThreadCommandType type;
|
|
char *path;
|
|
guint watch_id;
|
|
guint request;
|
|
} FAMThreadCommand;
|
|
|
|
typedef struct {
|
|
FAMThread thread_data;
|
|
guint event_id;
|
|
gboolean running;
|
|
GSList *watches;
|
|
} FAMWin32;
|
|
|
|
typedef struct {
|
|
gboolean error;
|
|
guint watch_id;
|
|
guint request;
|
|
guint code;
|
|
char *msg;
|
|
} FAMEvent;
|
|
|
|
static FAMWin32 *fam;
|
|
|
|
|
|
/****************************************************************************/
|
|
/* Watch thread
|
|
*/
|
|
|
|
static void
|
|
fam_event_free (FAMEvent *event)
|
|
{
|
|
if (event)
|
|
{
|
|
g_free (event->msg);
|
|
g_free (event);
|
|
}
|
|
}
|
|
|
|
static void
|
|
fam_thread_event (guint code,
|
|
gboolean error,
|
|
const char *msg,
|
|
guint watch_id,
|
|
guint request)
|
|
{
|
|
FAMEvent *event = g_new0 (FAMEvent, 1);
|
|
event->error = error;
|
|
event->code = code;
|
|
event->watch_id = watch_id;
|
|
event->request = request;
|
|
event->msg = g_strdup (msg);
|
|
_moo_event_queue_push (fam->event_id, event, (GDestroyNotify) fam_event_free);
|
|
}
|
|
|
|
static void
|
|
fam_thread_add_path (FAMThread *thr,
|
|
const char *path,
|
|
guint watch_id,
|
|
guint request)
|
|
{
|
|
gunichar2 *win_path;
|
|
|
|
if (thr->n_events >= MAXIMUM_WAIT_OBJECTS)
|
|
{
|
|
g_critical ("too many folders watched");
|
|
return;
|
|
}
|
|
|
|
win_path = g_utf8_to_utf16 (path, -1, NULL, NULL, NULL);
|
|
|
|
if (!win_path)
|
|
{
|
|
char *msg = g_strdup_printf ("Could not convert filename '%s' to UTF16",
|
|
path);
|
|
fam_thread_event (MOO_FILE_WATCH_ERROR_BAD_FILENAME, TRUE, msg, watch_id, request);
|
|
g_free (msg);
|
|
return;
|
|
}
|
|
|
|
thr->events[thr->n_events] =
|
|
FindFirstChangeNotificationW ((const WCHAR*) win_path, FALSE,
|
|
FILE_NOTIFY_CHANGE_FILE_NAME |
|
|
FILE_NOTIFY_CHANGE_DIR_NAME);
|
|
|
|
if (thr->events[thr->n_events] == INVALID_HANDLE_VALUE)
|
|
{
|
|
char *err = g_win32_error_message (GetLastError ());
|
|
char *msg = g_strdup_printf ("Could not convert start watch for '%s': %s", path, err);
|
|
fam_thread_event (MOO_FILE_WATCH_ERROR_FAILED, TRUE, msg, watch_id, request);
|
|
g_free (msg);
|
|
g_free (err);
|
|
return;
|
|
}
|
|
|
|
thr->watches[thr->n_events].watch_id = watch_id;
|
|
thr->watches[thr->n_events].request = request;
|
|
thr->watches[thr->n_events].path = g_strdup (path);
|
|
|
|
thr->n_events += 1;
|
|
}
|
|
|
|
static void
|
|
fam_thread_remove_path (FAMThread *thr,
|
|
guint watch_id,
|
|
guint request)
|
|
{
|
|
guint i;
|
|
|
|
for (i = 1; i < thr->n_events; ++i)
|
|
{
|
|
if (thr->watches[i].watch_id == watch_id &&
|
|
thr->watches[i].request == request)
|
|
{
|
|
FindCloseChangeNotification (thr->events[i]);
|
|
g_free (thr->watches[i].path);
|
|
|
|
if (i < thr->n_events - 1)
|
|
{
|
|
MOO_ELMMOVE (thr->watches + i,
|
|
thr->watches + i + 1,
|
|
thr->n_events - i - 1);
|
|
MOO_ELMMOVE (thr->events + i,
|
|
thr->events + i + 1,
|
|
thr->n_events - i - 1);
|
|
}
|
|
|
|
thr->n_events -= 1;
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
g_critical ("can't remove watch %d, request %d",
|
|
watch_id, request);
|
|
}
|
|
|
|
static void
|
|
fam_thread_check_dir (FAMThread *thr,
|
|
guint idx)
|
|
{
|
|
MgwStatBuf buf;
|
|
mgw_errno_t err;
|
|
|
|
if (mgw_stat (thr->watches[idx].path, &buf, &err) != 0 && err.value == MGW_ENOENT)
|
|
{
|
|
fam_thread_event (MOO_FILE_EVENT_DELETED,
|
|
FALSE, NULL,
|
|
thr->watches[idx].watch_id,
|
|
thr->watches[idx].request);
|
|
fam_thread_remove_path (thr,
|
|
thr->watches[idx].watch_id,
|
|
thr->watches[idx].request);
|
|
}
|
|
else
|
|
{
|
|
fam_thread_event (MOO_FILE_EVENT_CHANGED,
|
|
FALSE, NULL,
|
|
thr->watches[idx].watch_id,
|
|
thr->watches[idx].request);
|
|
|
|
if (!FindNextChangeNotification (thr->events[idx]))
|
|
{
|
|
gstr win_msg = gstr::wrap_new(g_win32_error_message(GetLastError()));
|
|
gstr msg = gstr::wrap_new(g_strdup_printf("Error in FindNextChangeNotification: %s", win_msg.get()));
|
|
|
|
fam_thread_event (MOO_FILE_WATCH_ERROR_FAILED, TRUE, msg,
|
|
thr->watches[idx].watch_id,
|
|
thr->watches[idx].request);
|
|
fam_thread_remove_path (thr,
|
|
thr->watches[idx].watch_id,
|
|
thr->watches[idx].request);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
fam_thread_do_command (FAMThread *thr)
|
|
{
|
|
FAMThreadCommand *cmd;
|
|
GSList *list = NULL;
|
|
|
|
DEBUG_PRINT ("fam_thread_do_command start");
|
|
|
|
g_mutex_lock (thr->lock);
|
|
while ((cmd = (FAMThreadCommand*) g_async_queue_try_pop (thr->incoming)))
|
|
list = g_slist_prepend (list, cmd);
|
|
ResetEvent (thr->events[0]);
|
|
g_mutex_unlock (thr->lock);
|
|
|
|
list = g_slist_reverse (list);
|
|
|
|
while (list)
|
|
{
|
|
cmd = (FAMThreadCommand*) list->data;
|
|
|
|
switch (cmd->type)
|
|
{
|
|
case COMMAND_ADD_PATH:
|
|
fam_thread_add_path (thr, cmd->path, cmd->watch_id, cmd->request);
|
|
break;
|
|
|
|
case COMMAND_REMOVE_PATH:
|
|
fam_thread_remove_path (thr, cmd->watch_id, cmd->request);
|
|
break;
|
|
}
|
|
|
|
list = g_slist_delete_link (list, list);
|
|
g_free (cmd->path);
|
|
g_free (cmd);
|
|
}
|
|
|
|
DEBUG_PRINT ("fam_thread_do_command end");
|
|
}
|
|
|
|
|
|
static gpointer
|
|
fam_thread_main (FAMThread *thr)
|
|
{
|
|
while (TRUE)
|
|
{
|
|
int ret;
|
|
|
|
DEBUG_PRINT ("calling WaitForMultipleObjects for %d handles", thr->n_events);
|
|
ret = WaitForMultipleObjects (thr->n_events, thr->events, FALSE, INFINITE);
|
|
DEBUG_PRINT ("WaitForMultipleObjects returned %d", ret);
|
|
|
|
if (ret == (int) WAIT_FAILED)
|
|
{
|
|
char *msg = g_win32_error_message (GetLastError ());
|
|
g_critical ("%s", msg);
|
|
g_free (msg);
|
|
break;
|
|
}
|
|
|
|
if (ret == WAIT_OBJECT_0)
|
|
fam_thread_do_command (thr);
|
|
else if ((int) WAIT_OBJECT_0 < ret && ret < (int) thr->n_events)
|
|
fam_thread_check_dir (thr, ret - WAIT_OBJECT_0);
|
|
else
|
|
{
|
|
g_critical ("oops");
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* XXX cleanup */
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
fam_thread_command (FAMThreadCommandType type,
|
|
const char *filename,
|
|
guint watch_id,
|
|
guint request)
|
|
{
|
|
FAMThreadCommand *cmd;
|
|
|
|
cmd = g_new0 (FAMThreadCommand, 1);
|
|
cmd->type = type;
|
|
cmd->path = g_strdup (filename);
|
|
cmd->watch_id = watch_id;
|
|
cmd->request = request;
|
|
|
|
g_mutex_lock (fam->thread_data.lock);
|
|
g_async_queue_push (fam->thread_data.incoming, cmd);
|
|
SetEvent (fam->thread_data.events[0]);
|
|
g_mutex_unlock (fam->thread_data.lock);
|
|
}
|
|
|
|
/****************************************************************************/
|
|
/* Monitors
|
|
*/
|
|
|
|
static MooFileWatch *
|
|
find_watch (guint id)
|
|
{
|
|
GSList *l;
|
|
|
|
for (l = fam->watches; l != NULL; l = l->next)
|
|
{
|
|
MooFileWatch *watch = (MooFileWatch*) l->data;
|
|
|
|
if (watch->impl().id == id)
|
|
return watch;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
do_event (FAMEvent *event)
|
|
{
|
|
MooFileWatch *watch;
|
|
Monitor *monitor;
|
|
MooFileEvent watch_event;
|
|
|
|
watch = find_watch (event->watch_id);
|
|
|
|
if (!watch)
|
|
{
|
|
DEBUG_PRINT ("got event for dead watch %d", event->watch_id);
|
|
return;
|
|
}
|
|
|
|
monitor = (Monitor*)
|
|
g_hash_table_lookup (watch->impl().requests, GUINT_TO_POINTER (event->request));
|
|
|
|
if (!monitor)
|
|
{
|
|
DEBUG_PRINT ("got event for dead monitor %d", event->request);
|
|
return;
|
|
}
|
|
|
|
watch_event.monitor_id = event->request;
|
|
watch_event.filename = monitor->filename;
|
|
watch_event.error = NULL;
|
|
|
|
if (event->error)
|
|
{
|
|
watch_event.code = MOO_FILE_EVENT_ERROR;
|
|
g_set_error (&watch_event.error, MOO_FILE_WATCH_ERROR,
|
|
event->code,
|
|
event->msg ? event->msg : "FAILED");
|
|
DEBUG_PRINT ("got error for watch %d: %s", event->watch_id, watch_event.error->message);
|
|
}
|
|
else
|
|
{
|
|
DEBUG_PRINT ("got event for filename %s", monitor->filename);
|
|
watch_event.code = (MooFileEventCode) event->code;
|
|
}
|
|
|
|
moo_file_watch_emit_event (watch, &watch_event, monitor);
|
|
|
|
if (watch_event.error)
|
|
g_error_free (watch_event.error);
|
|
}
|
|
|
|
static void
|
|
event_callback (GList *events)
|
|
{
|
|
GList *trimmed = NULL;
|
|
|
|
while (events)
|
|
{
|
|
GList *l;
|
|
FAMEvent *event = (FAMEvent*) events->data;
|
|
gboolean found = FALSE;
|
|
|
|
event = (FAMEvent*) events->data;
|
|
events = events->next;
|
|
|
|
for (l = trimmed; l != NULL; l = l->next)
|
|
{
|
|
FAMEvent *old_event = (FAMEvent*) l->data;
|
|
|
|
if (old_event->watch_id == event->watch_id &&
|
|
old_event->request == event->request)
|
|
{
|
|
found = TRUE;
|
|
|
|
if (!old_event->error && event->error)
|
|
l->data = event;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found)
|
|
trimmed = g_list_prepend (trimmed, event);
|
|
}
|
|
|
|
trimmed = g_list_reverse (trimmed);
|
|
|
|
while (trimmed)
|
|
{
|
|
do_event ((FAMEvent*) trimmed->data);
|
|
trimmed = g_list_delete_link (trimmed, trimmed);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
fam_win32_init (void)
|
|
{
|
|
if (!fam)
|
|
{
|
|
GError *error = NULL;
|
|
|
|
fam = g_new0 (FAMWin32, 1);
|
|
fam->running = FALSE;
|
|
|
|
fam->thread_data.n_events = 1;
|
|
fam->thread_data.events[0] = CreateEvent (NULL, TRUE, FALSE, NULL);
|
|
|
|
if (!fam->thread_data.events[0])
|
|
{
|
|
char *msg = g_win32_error_message (GetLastError ());
|
|
g_critical ("could not create incoming event: %s", msg);
|
|
g_free (msg);
|
|
return FALSE;
|
|
}
|
|
|
|
fam->thread_data.incoming = g_async_queue_new ();
|
|
fam->thread_data.lock = g_mutex_new ();
|
|
fam->event_id = _moo_event_queue_connect ((MooEventQueueCallback) event_callback,
|
|
NULL, NULL);
|
|
|
|
if (!g_thread_create ((GThreadFunc) fam_thread_main, &fam->thread_data, FALSE, &error))
|
|
{
|
|
g_critical ("could not start watch thread: %s", moo_error_message (error));
|
|
g_error_free (error);
|
|
}
|
|
else
|
|
{
|
|
fam->running = TRUE;
|
|
DEBUG_PRINT ("initialized folder watch");
|
|
}
|
|
}
|
|
|
|
return fam->running;
|
|
}
|
|
|
|
static gboolean
|
|
watch_win32_start(MooFileWatch& watch,
|
|
GError** error)
|
|
{
|
|
if (!watch_stat_start (watch, error))
|
|
return FALSE;
|
|
|
|
if (!fam_win32_init ())
|
|
{
|
|
g_set_error (error, MOO_FILE_WATCH_ERROR,
|
|
MOO_FILE_WATCH_ERROR_FAILED,
|
|
"Could not initialize folder watch thread");
|
|
watch_stat_shutdown(watch, NULL);
|
|
return FALSE;
|
|
}
|
|
|
|
fam->watches = g_slist_prepend (fam->watches, &watch);
|
|
DEBUG_PRINT ("started watch %d", watch.impl().id);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
watch_win32_shutdown(MooFileWatch& watch,
|
|
GError** error)
|
|
{
|
|
return watch_stat_shutdown(watch, error);
|
|
}
|
|
|
|
|
|
static gboolean
|
|
watch_win32_start_monitor (MooFileWatch& watch,
|
|
Monitor* monitor,
|
|
GError** error)
|
|
{
|
|
MgwStatBuf buf;
|
|
mgw_errno_t err;
|
|
|
|
g_return_val_if_fail (monitor->filename != NULL, FALSE);
|
|
|
|
if (mgw_stat (monitor->filename, &buf, &err) != 0)
|
|
{
|
|
g_set_error (error, MOO_FILE_WATCH_ERROR,
|
|
errno_to_file_error (err),
|
|
"stat: %s", mgw_strerror (err));
|
|
return FALSE;
|
|
}
|
|
|
|
monitor->isdir = buf.isdir;
|
|
|
|
if (!monitor->isdir)
|
|
return watch_stat_start_monitor(watch, monitor, error);
|
|
|
|
DEBUG_PRINT ("created monitor for '%s'", monitor->filename);
|
|
|
|
monitor->id = get_new_monitor_id ();
|
|
|
|
fam_thread_command(COMMAND_ADD_PATH, monitor->filename, watch.impl().id, monitor->id);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void
|
|
watch_win32_stop_monitor (MooFileWatch& watch,
|
|
Monitor& monitor)
|
|
{
|
|
if (monitor.isdir)
|
|
{
|
|
DEBUG_PRINT ("removing monitor for '%s'", monitor.filename);
|
|
fam_thread_command (COMMAND_REMOVE_PATH, monitor.filename,
|
|
watch.impl().id, monitor.id);
|
|
}
|
|
}
|
|
#endif /* __WIN32__ */
|