• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /* GIO - GLib Input, Output and Streaming Library
2  *
3  * Copyright (C) 2006-2007 Red Hat, Inc.
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2.1 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General
16  * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
17  *
18  * Author: Alexander Larsson <alexl@redhat.com>
19  */
20 
21 #include "config.h"
22 
23 #include "gioenumtypes.h"
24 #include "glocalfilemonitor.h"
25 #include "giomodule-priv.h"
26 #include "gioerror.h"
27 #include "glibintl.h"
28 #include "glocalfile.h"
29 #include "glib-private.h"
30 
31 #include <string.h>
32 
33 #define DEFAULT_RATE_LIMIT                           800 * G_TIME_SPAN_MILLISECOND
34 #define VIRTUAL_CHANGES_DONE_DELAY                     2 * G_TIME_SPAN_SECOND
35 
36 /* GFileMonitorSource is a GSource responsible for emitting the changed
37  * signals in the owner-context of the GFileMonitor.
38  *
39  * It contains functionality for cross-thread queuing of events.  It
40  * also handles merging of CHANGED events and emission of CHANGES_DONE
41  * events.
42  *
43  * We use the "priv" pointer in the external struct to store it.
44  */
45 struct _GFileMonitorSource {
46   GSource       source;
47 
48   GMutex        lock;
49   GWeakRef      instance_ref;
50   GFileMonitorFlags flags;
51   gchar        *dirname;
52   gchar        *basename;
53   gchar        *filename;
54   GSequence    *pending_changes; /* sorted by ready time */
55   GHashTable   *pending_changes_table;
56   GQueue        event_queue;
57   gint64        rate_limit;
58 };
59 
60 /* PendingChange is a struct to keep track of a file that needs to have
61  * (at least) a CHANGES_DONE_HINT event sent for it in the near future.
62  *
63  * If 'dirty' is TRUE then a CHANGED event also needs to be sent.
64  *
65  * last_emission is the last time a CHANGED event was emitted.  It is
66  * used to calculate the time to send the next event.
67  */
68 typedef struct {
69   gchar    *child;
70   guint64   last_emission : 63;
71   guint64   dirty         :  1;
72 } PendingChange;
73 
74 /* QueuedEvent is a signal that will be sent immediately, as soon as the
75  * source gets a chance to dispatch.  The existence of any queued event
76  * implies that the source is ready now.
77  */
78 typedef struct
79 {
80   GFileMonitorEvent event_type;
81   GFile *child;
82   GFile *other;
83 } QueuedEvent;
84 
85 static gint64
pending_change_get_ready_time(const PendingChange * change,GFileMonitorSource * fms)86 pending_change_get_ready_time (const PendingChange *change,
87                                GFileMonitorSource  *fms)
88 {
89   if (change->dirty)
90     return change->last_emission + fms->rate_limit;
91   else
92     return change->last_emission + VIRTUAL_CHANGES_DONE_DELAY;
93 }
94 
95 static int
pending_change_compare_ready_time(gconstpointer a_p,gconstpointer b_p,gpointer user_data)96 pending_change_compare_ready_time (gconstpointer a_p,
97                                    gconstpointer b_p,
98                                    gpointer      user_data)
99 {
100   GFileMonitorSource *fms = user_data;
101   const PendingChange *a = a_p;
102   const PendingChange *b = b_p;
103   gint64 ready_time_a;
104   gint64 ready_time_b;
105 
106   ready_time_a = pending_change_get_ready_time (a, fms);
107   ready_time_b = pending_change_get_ready_time (b, fms);
108 
109   if (ready_time_a < ready_time_b)
110     return -1;
111   else
112     return ready_time_a > ready_time_b;
113 }
114 
115 static void
pending_change_free(gpointer data)116 pending_change_free (gpointer data)
117 {
118   PendingChange *change = data;
119 
120   g_free (change->child);
121 
122   g_slice_free (PendingChange, change);
123 }
124 
125 static void
queued_event_free(QueuedEvent * event)126 queued_event_free (QueuedEvent *event)
127 {
128   g_object_unref (event->child);
129   if (event->other)
130     g_object_unref (event->other);
131 
132   g_slice_free (QueuedEvent, event);
133 }
134 
135 static gint64
g_file_monitor_source_get_ready_time(GFileMonitorSource * fms)136 g_file_monitor_source_get_ready_time (GFileMonitorSource *fms)
137 {
138   GSequenceIter *iter;
139 
140   if (fms->event_queue.length)
141     return 0;
142 
143   iter = g_sequence_get_begin_iter (fms->pending_changes);
144   if (g_sequence_iter_is_end (iter))
145     return -1;
146 
147   return pending_change_get_ready_time (g_sequence_get (iter), fms);
148 }
149 
150 static void
g_file_monitor_source_update_ready_time(GFileMonitorSource * fms)151 g_file_monitor_source_update_ready_time (GFileMonitorSource *fms)
152 {
153   g_source_set_ready_time ((GSource *) fms, g_file_monitor_source_get_ready_time (fms));
154 }
155 
156 static GSequenceIter *
g_file_monitor_source_find_pending_change(GFileMonitorSource * fms,const gchar * child)157 g_file_monitor_source_find_pending_change (GFileMonitorSource *fms,
158                                            const gchar        *child)
159 {
160   return g_hash_table_lookup (fms->pending_changes_table, child);
161 }
162 
163 static void
g_file_monitor_source_add_pending_change(GFileMonitorSource * fms,const gchar * child,gint64 now)164 g_file_monitor_source_add_pending_change (GFileMonitorSource *fms,
165                                           const gchar        *child,
166                                           gint64              now)
167 {
168   PendingChange *change;
169   GSequenceIter *iter;
170 
171   change = g_slice_new (PendingChange);
172   change->child = g_strdup (child);
173   change->last_emission = now;
174   change->dirty = FALSE;
175 
176   iter = g_sequence_insert_sorted (fms->pending_changes, change, pending_change_compare_ready_time, fms);
177   g_hash_table_insert (fms->pending_changes_table, change->child, iter);
178 }
179 
180 static gboolean
g_file_monitor_source_set_pending_change_dirty(GFileMonitorSource * fms,GSequenceIter * iter)181 g_file_monitor_source_set_pending_change_dirty (GFileMonitorSource *fms,
182                                                 GSequenceIter      *iter)
183 {
184   PendingChange *change;
185 
186   change = g_sequence_get (iter);
187 
188   /* if it was already dirty then this change is 'uninteresting' */
189   if (change->dirty)
190     return FALSE;
191 
192   change->dirty = TRUE;
193 
194   g_sequence_sort_changed (iter, pending_change_compare_ready_time, fms);
195 
196   return TRUE;
197 }
198 
199 static gboolean
g_file_monitor_source_get_pending_change_dirty(GFileMonitorSource * fms,GSequenceIter * iter)200 g_file_monitor_source_get_pending_change_dirty (GFileMonitorSource *fms,
201                                                 GSequenceIter      *iter)
202 {
203   PendingChange *change;
204 
205   change = g_sequence_get (iter);
206 
207   return change->dirty;
208 }
209 
210 static void
g_file_monitor_source_remove_pending_change(GFileMonitorSource * fms,GSequenceIter * iter,const gchar * child)211 g_file_monitor_source_remove_pending_change (GFileMonitorSource *fms,
212                                              GSequenceIter      *iter,
213                                              const gchar        *child)
214 {
215   /* must remove the hash entry first -- its key is owned by the data
216    * which will be freed when removing the sequence iter
217    */
218   g_hash_table_remove (fms->pending_changes_table, child);
219   g_sequence_remove (iter);
220 }
221 
222 static void
g_file_monitor_source_queue_event(GFileMonitorSource * fms,GFileMonitorEvent event_type,const gchar * child,GFile * other)223 g_file_monitor_source_queue_event (GFileMonitorSource *fms,
224                                    GFileMonitorEvent   event_type,
225                                    const gchar        *child,
226                                    GFile              *other)
227 {
228   QueuedEvent *event;
229 
230   event = g_slice_new (QueuedEvent);
231   event->event_type = event_type;
232   if (child != NULL && fms->dirname != NULL)
233     event->child = g_local_file_new_from_dirname_and_basename (fms->dirname, child);
234   else if (child != NULL)
235     {
236       gchar *dirname = g_path_get_dirname (fms->filename);
237       event->child = g_local_file_new_from_dirname_and_basename (dirname, child);
238       g_free (dirname);
239     }
240   else if (fms->dirname)
241     event->child = _g_local_file_new (fms->dirname);
242   else if (fms->filename)
243     event->child = _g_local_file_new (fms->filename);
244   event->other = other;
245   if (other)
246     g_object_ref (other);
247 
248   g_queue_push_tail (&fms->event_queue, event);
249 }
250 
251 static gboolean
g_file_monitor_source_file_changed(GFileMonitorSource * fms,const gchar * child,gint64 now)252 g_file_monitor_source_file_changed (GFileMonitorSource *fms,
253                                     const gchar        *child,
254                                     gint64              now)
255 {
256   GSequenceIter *pending;
257   gboolean interesting;
258 
259   pending = g_file_monitor_source_find_pending_change (fms, child);
260 
261   /* If there is no pending change, emit one and create a record,
262    * else: just mark the existing record as dirty.
263    */
264   if (!pending)
265     {
266       g_file_monitor_source_queue_event (fms, G_FILE_MONITOR_EVENT_CHANGED, child, NULL);
267       g_file_monitor_source_add_pending_change (fms, child, now);
268       interesting = TRUE;
269     }
270   else
271     interesting = g_file_monitor_source_set_pending_change_dirty (fms, pending);
272 
273   g_file_monitor_source_update_ready_time (fms);
274 
275   return interesting;
276 }
277 
278 static void
g_file_monitor_source_file_changes_done(GFileMonitorSource * fms,const gchar * child)279 g_file_monitor_source_file_changes_done (GFileMonitorSource *fms,
280                                          const gchar        *child)
281 {
282   GSequenceIter *pending;
283 
284   pending = g_file_monitor_source_find_pending_change (fms, child);
285   if (pending)
286     {
287       /* If it is dirty, make sure we push out the last CHANGED event */
288       if (g_file_monitor_source_get_pending_change_dirty (fms, pending))
289         g_file_monitor_source_queue_event (fms, G_FILE_MONITOR_EVENT_CHANGED, child, NULL);
290 
291       g_file_monitor_source_queue_event (fms, G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, child, NULL);
292       g_file_monitor_source_remove_pending_change (fms, pending, child);
293     }
294 }
295 
296 static void
g_file_monitor_source_file_created(GFileMonitorSource * fms,const gchar * child,gint64 event_time)297 g_file_monitor_source_file_created (GFileMonitorSource *fms,
298                                     const gchar        *child,
299                                     gint64              event_time)
300 {
301   /* Unlikely, but if we have pending changes for this filename, make
302    * sure we flush those out first, before creating the new ones.
303    */
304   g_file_monitor_source_file_changes_done (fms, child);
305 
306   /* Emit CREATE and add a pending changes record */
307   g_file_monitor_source_queue_event (fms, G_FILE_MONITOR_EVENT_CREATED, child, NULL);
308   g_file_monitor_source_add_pending_change (fms, child, event_time);
309 }
310 
311 static void
g_file_monitor_source_send_event(GFileMonitorSource * fms,GFileMonitorEvent event_type,const gchar * child,GFile * other)312 g_file_monitor_source_send_event (GFileMonitorSource *fms,
313                                   GFileMonitorEvent   event_type,
314                                   const gchar        *child,
315                                   GFile              *other)
316 {
317   /* always flush any pending changes before we queue a new event */
318   g_file_monitor_source_file_changes_done (fms, child);
319   g_file_monitor_source_queue_event (fms, event_type, child, other);
320 }
321 
322 static void
g_file_monitor_source_send_synthetic_created(GFileMonitorSource * fms,const gchar * child)323 g_file_monitor_source_send_synthetic_created (GFileMonitorSource *fms,
324                                               const gchar        *child)
325 {
326   g_file_monitor_source_file_changes_done (fms, child);
327   g_file_monitor_source_queue_event (fms, G_FILE_MONITOR_EVENT_CREATED, child, NULL);
328   g_file_monitor_source_queue_event (fms, G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, child, NULL);
329 }
330 
331 #ifndef G_DISABLE_ASSERT
332 static gboolean
is_basename(const gchar * name)333 is_basename (const gchar *name)
334 {
335   if (name[0] == '.' && ((name[1] == '.' && name[2] == '\0') || name[1] == '\0'))
336     return FALSE;
337 
338   return !strchr (name, '/');
339 }
340 #endif  /* !G_DISABLE_ASSERT */
341 
342 gboolean
g_file_monitor_source_handle_event(GFileMonitorSource * fms,GFileMonitorEvent event_type,const gchar * child,const gchar * rename_to,GFile * other,gint64 event_time)343 g_file_monitor_source_handle_event (GFileMonitorSource *fms,
344                                     GFileMonitorEvent   event_type,
345                                     const gchar        *child,
346                                     const gchar        *rename_to,
347                                     GFile              *other,
348                                     gint64              event_time)
349 {
350   gboolean interesting = TRUE;
351   GFileMonitor *instance = NULL;
352 
353   g_assert (!child || is_basename (child));
354   g_assert (!rename_to || is_basename (rename_to));
355 
356   if (fms->basename && (!child || !g_str_equal (child, fms->basename))
357                     && (!rename_to || !g_str_equal (rename_to, fms->basename)))
358     return TRUE;
359 
360   g_mutex_lock (&fms->lock);
361 
362   /* monitor is already gone -- don't bother */
363   instance = g_weak_ref_get (&fms->instance_ref);
364   if (instance == NULL)
365     {
366       g_mutex_unlock (&fms->lock);
367       return TRUE;
368     }
369 
370   switch (event_type)
371     {
372     case G_FILE_MONITOR_EVENT_CREATED:
373       g_assert (!other && !rename_to);
374       g_file_monitor_source_file_created (fms, child, event_time);
375       break;
376 
377     case G_FILE_MONITOR_EVENT_CHANGED:
378       g_assert (!other && !rename_to);
379       interesting = g_file_monitor_source_file_changed (fms, child, event_time);
380       break;
381 
382     case G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT:
383       g_assert (!other && !rename_to);
384       g_file_monitor_source_file_changes_done (fms, child);
385       break;
386 
387     case G_FILE_MONITOR_EVENT_MOVED_IN:
388       g_assert (!rename_to);
389       if (fms->flags & G_FILE_MONITOR_WATCH_MOVES)
390         g_file_monitor_source_send_event (fms, G_FILE_MONITOR_EVENT_MOVED_IN, child, other);
391       else
392         g_file_monitor_source_send_synthetic_created (fms, child);
393       break;
394 
395     case G_FILE_MONITOR_EVENT_MOVED_OUT:
396       g_assert (!rename_to);
397       if (fms->flags & G_FILE_MONITOR_WATCH_MOVES)
398         g_file_monitor_source_send_event (fms, G_FILE_MONITOR_EVENT_MOVED_OUT, child, other);
399       else if (other && (fms->flags & G_FILE_MONITOR_SEND_MOVED))
400         g_file_monitor_source_send_event (fms, G_FILE_MONITOR_EVENT_MOVED, child, other);
401       else
402         g_file_monitor_source_send_event (fms, G_FILE_MONITOR_EVENT_DELETED, child, NULL);
403       break;
404 
405     case G_FILE_MONITOR_EVENT_RENAMED:
406       g_assert (!other && rename_to);
407       if (fms->flags & (G_FILE_MONITOR_WATCH_MOVES | G_FILE_MONITOR_SEND_MOVED))
408         {
409           GFile *other;
410           const gchar *dirname;
411           gchar *allocated_dirname = NULL;
412           GFileMonitorEvent event;
413 
414           event = (fms->flags & G_FILE_MONITOR_WATCH_MOVES) ? G_FILE_MONITOR_EVENT_RENAMED : G_FILE_MONITOR_EVENT_MOVED;
415 
416           if (fms->dirname != NULL)
417             dirname = fms->dirname;
418           else
419             {
420               allocated_dirname = g_path_get_dirname (fms->filename);
421               dirname = allocated_dirname;
422             }
423 
424           other = g_local_file_new_from_dirname_and_basename (dirname, rename_to);
425           g_file_monitor_source_file_changes_done (fms, rename_to);
426           g_file_monitor_source_send_event (fms, event, child, other);
427 
428           g_object_unref (other);
429           g_free (allocated_dirname);
430         }
431       else
432         {
433           g_file_monitor_source_send_event (fms, G_FILE_MONITOR_EVENT_DELETED, child, NULL);
434           g_file_monitor_source_send_synthetic_created (fms, rename_to);
435         }
436       break;
437 
438     case G_FILE_MONITOR_EVENT_DELETED:
439     case G_FILE_MONITOR_EVENT_ATTRIBUTE_CHANGED:
440     case G_FILE_MONITOR_EVENT_PRE_UNMOUNT:
441     case G_FILE_MONITOR_EVENT_UNMOUNTED:
442       g_assert (!other && !rename_to);
443       g_file_monitor_source_send_event (fms, event_type, child, NULL);
444       break;
445 
446     case G_FILE_MONITOR_EVENT_MOVED:
447       /* was never available in this API */
448     default:
449       g_assert_not_reached ();
450     }
451 
452   g_file_monitor_source_update_ready_time (fms);
453 
454   g_mutex_unlock (&fms->lock);
455   g_clear_object (&instance);
456 
457   return interesting;
458 }
459 
460 static gint64
g_file_monitor_source_get_rate_limit(GFileMonitorSource * fms)461 g_file_monitor_source_get_rate_limit (GFileMonitorSource *fms)
462 {
463   gint64 rate_limit;
464 
465   g_mutex_lock (&fms->lock);
466   rate_limit = fms->rate_limit;
467   g_mutex_unlock (&fms->lock);
468 
469   return rate_limit;
470 }
471 
472 static gboolean
g_file_monitor_source_set_rate_limit(GFileMonitorSource * fms,gint64 rate_limit)473 g_file_monitor_source_set_rate_limit (GFileMonitorSource *fms,
474                                       gint64              rate_limit)
475 {
476   gboolean changed;
477 
478   g_mutex_lock (&fms->lock);
479 
480   if (rate_limit != fms->rate_limit)
481     {
482       fms->rate_limit = rate_limit;
483 
484       g_sequence_sort (fms->pending_changes, pending_change_compare_ready_time, fms);
485       g_file_monitor_source_update_ready_time (fms);
486 
487       changed = TRUE;
488     }
489   else
490     changed = FALSE;
491 
492   g_mutex_unlock (&fms->lock);
493 
494   return changed;
495 }
496 
497 static gboolean
g_file_monitor_source_dispatch(GSource * source,GSourceFunc callback,gpointer user_data)498 g_file_monitor_source_dispatch (GSource     *source,
499                                 GSourceFunc  callback,
500                                 gpointer     user_data)
501 {
502   GFileMonitorSource *fms = (GFileMonitorSource *) source;
503   QueuedEvent *event;
504   GQueue event_queue;
505   gint64 now;
506   GFileMonitor *instance = NULL;
507 
508   /* make sure the monitor still exists */
509   instance = g_weak_ref_get (&fms->instance_ref);
510   if (instance == NULL)
511     return FALSE;
512 
513   now = g_source_get_time (source);
514 
515   /* Acquire the lock once and grab all events in one go, handling the
516    * queued events first.  This avoids strange possibilities in cases of
517    * long delays, such as CHANGED events coming before CREATED events.
518    *
519    * We do this by converting the applicable pending changes into queued
520    * events (after the ones already queued) and then stealing the entire
521    * event queue in one go.
522    */
523   g_mutex_lock (&fms->lock);
524 
525   /* Create events for any pending changes that are due to fire */
526   while (!g_sequence_is_empty (fms->pending_changes))
527     {
528       GSequenceIter *iter = g_sequence_get_begin_iter (fms->pending_changes);
529       PendingChange *pending = g_sequence_get (iter);
530 
531       /* We've gotten to a pending change that's not ready.  Stop. */
532       if (pending_change_get_ready_time (pending, fms) > now)
533         break;
534 
535       if (pending->dirty)
536         {
537           /* It's time to send another CHANGED and update the record */
538           g_file_monitor_source_queue_event (fms, G_FILE_MONITOR_EVENT_CHANGED, pending->child, NULL);
539           pending->last_emission = now;
540           pending->dirty = FALSE;
541 
542           g_sequence_sort_changed (iter, pending_change_compare_ready_time, fms);
543         }
544       else
545         {
546           /* It's time to send CHANGES_DONE and remove the pending record */
547           g_file_monitor_source_queue_event (fms, G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, pending->child, NULL);
548           g_file_monitor_source_remove_pending_change (fms, iter, pending->child);
549         }
550     }
551 
552   /* Steal the queue */
553   memcpy (&event_queue, &fms->event_queue, sizeof event_queue);
554   memset (&fms->event_queue, 0, sizeof fms->event_queue);
555 
556   g_file_monitor_source_update_ready_time (fms);
557 
558   g_clear_object (&instance);
559   g_mutex_unlock (&fms->lock);
560 
561   /* We now have our list of events to deliver */
562   while ((event = g_queue_pop_head (&event_queue)))
563     {
564       /* an event handler could destroy 'instance', so check each time */
565       instance = g_weak_ref_get (&fms->instance_ref);
566       if (instance != NULL)
567         g_file_monitor_emit_event (instance, event->child, event->other, event->event_type);
568 
569       g_clear_object (&instance);
570       queued_event_free (event);
571     }
572 
573   return TRUE;
574 }
575 
576 static void
g_file_monitor_source_dispose(GFileMonitorSource * fms)577 g_file_monitor_source_dispose (GFileMonitorSource *fms)
578 {
579   GHashTableIter iter;
580   gpointer seqiter;
581   QueuedEvent *event;
582 
583   g_mutex_lock (&fms->lock);
584 
585   g_hash_table_iter_init (&iter, fms->pending_changes_table);
586   while (g_hash_table_iter_next (&iter, NULL, &seqiter))
587     {
588       g_hash_table_iter_remove (&iter);
589       g_sequence_remove (seqiter);
590     }
591 
592   while ((event = g_queue_pop_head (&fms->event_queue)))
593     queued_event_free (event);
594 
595   g_assert (g_sequence_is_empty (fms->pending_changes));
596   g_assert (g_hash_table_size (fms->pending_changes_table) == 0);
597   g_assert (fms->event_queue.length == 0);
598   g_weak_ref_set (&fms->instance_ref, NULL);
599 
600   g_file_monitor_source_update_ready_time (fms);
601 
602   g_mutex_unlock (&fms->lock);
603 
604   g_source_destroy ((GSource *) fms);
605 }
606 
607 static void
g_file_monitor_source_finalize(GSource * source)608 g_file_monitor_source_finalize (GSource *source)
609 {
610   GFileMonitorSource *fms = (GFileMonitorSource *) source;
611 
612   /* should already have been cleared in dispose of the monitor */
613   g_assert (g_weak_ref_get (&fms->instance_ref) == NULL);
614   g_weak_ref_clear (&fms->instance_ref);
615 
616   g_assert (g_sequence_is_empty (fms->pending_changes));
617   g_assert (g_hash_table_size (fms->pending_changes_table) == 0);
618   g_assert (fms->event_queue.length == 0);
619 
620   g_hash_table_unref (fms->pending_changes_table);
621   g_sequence_free (fms->pending_changes);
622 
623   g_free (fms->dirname);
624   g_free (fms->basename);
625   g_free (fms->filename);
626 
627   g_mutex_clear (&fms->lock);
628 }
629 
630 static guint
str_hash0(gconstpointer str)631 str_hash0 (gconstpointer str)
632 {
633   return str ? g_str_hash (str) : 0;
634 }
635 
636 static gboolean
str_equal0(gconstpointer a,gconstpointer b)637 str_equal0 (gconstpointer a,
638             gconstpointer b)
639 {
640   return g_strcmp0 (a, b) == 0;
641 }
642 
643 static GFileMonitorSource *
g_file_monitor_source_new(gpointer instance,const gchar * filename,gboolean is_directory,GFileMonitorFlags flags)644 g_file_monitor_source_new (gpointer           instance,
645                            const gchar       *filename,
646                            gboolean           is_directory,
647                            GFileMonitorFlags  flags)
648 {
649   static GSourceFuncs source_funcs = {
650     NULL, NULL,
651     g_file_monitor_source_dispatch,
652     g_file_monitor_source_finalize,
653     NULL, NULL
654   };
655   GFileMonitorSource *fms;
656   GSource *source;
657 
658   source = g_source_new (&source_funcs, sizeof (GFileMonitorSource));
659   fms = (GFileMonitorSource *) source;
660 
661   g_source_set_name (source, "GFileMonitorSource");
662 
663   g_mutex_init (&fms->lock);
664   g_weak_ref_init (&fms->instance_ref, instance);
665   fms->pending_changes = g_sequence_new (pending_change_free);
666   fms->pending_changes_table = g_hash_table_new (str_hash0, str_equal0);
667   fms->rate_limit = DEFAULT_RATE_LIMIT;
668   fms->flags = flags;
669 
670   if (is_directory)
671     {
672       fms->dirname = g_strdup (filename);
673       fms->basename = NULL;
674       fms->filename = NULL;
675     }
676   else if (flags & G_FILE_MONITOR_WATCH_HARD_LINKS)
677     {
678       fms->dirname = NULL;
679       fms->basename = NULL;
680       fms->filename = g_strdup (filename);
681     }
682   else
683     {
684       fms->dirname = g_path_get_dirname (filename);
685       fms->basename = g_path_get_basename (filename);
686       fms->filename = NULL;
687     }
688 
689   return fms;
690 }
691 
692 G_DEFINE_ABSTRACT_TYPE (GLocalFileMonitor, g_local_file_monitor, G_TYPE_FILE_MONITOR)
693 
694 enum {
695   PROP_0,
696   PROP_RATE_LIMIT,
697 };
698 
699 static void
g_local_file_monitor_get_property(GObject * object,guint prop_id,GValue * value,GParamSpec * pspec)700 g_local_file_monitor_get_property (GObject *object, guint prop_id,
701                                    GValue *value, GParamSpec *pspec)
702 {
703   GLocalFileMonitor *monitor = G_LOCAL_FILE_MONITOR (object);
704   gint64 rate_limit;
705 
706   g_assert (prop_id == PROP_RATE_LIMIT);
707 
708   rate_limit = g_file_monitor_source_get_rate_limit (monitor->source);
709   rate_limit /= G_TIME_SPAN_MILLISECOND;
710 
711   g_value_set_int (value, rate_limit);
712 }
713 
714 static void
g_local_file_monitor_set_property(GObject * object,guint prop_id,const GValue * value,GParamSpec * pspec)715 g_local_file_monitor_set_property (GObject *object, guint prop_id,
716                                    const GValue *value, GParamSpec *pspec)
717 {
718   GLocalFileMonitor *monitor = G_LOCAL_FILE_MONITOR (object);
719   gint64 rate_limit;
720 
721   g_assert (prop_id == PROP_RATE_LIMIT);
722 
723   rate_limit = g_value_get_int (value);
724   rate_limit *= G_TIME_SPAN_MILLISECOND;
725 
726   if (g_file_monitor_source_set_rate_limit (monitor->source, rate_limit))
727     g_object_notify (object, "rate-limit");
728 }
729 
730 #ifndef G_OS_WIN32
731 static void
g_local_file_monitor_mounts_changed(GUnixMountMonitor * mount_monitor,gpointer user_data)732 g_local_file_monitor_mounts_changed (GUnixMountMonitor *mount_monitor,
733                                      gpointer           user_data)
734 {
735   GLocalFileMonitor *local_monitor = user_data;
736   GUnixMountEntry *mount;
737   gboolean is_mounted;
738   GFile *file;
739 
740   /* Emulate unmount detection */
741   mount = g_unix_mount_at (local_monitor->source->dirname, NULL);
742 
743   is_mounted = mount != NULL;
744 
745   if (mount)
746     g_unix_mount_free (mount);
747 
748   if (local_monitor->was_mounted != is_mounted)
749     {
750       if (local_monitor->was_mounted && !is_mounted)
751         {
752           file = g_file_new_for_path (local_monitor->source->dirname);
753           g_file_monitor_emit_event (G_FILE_MONITOR (local_monitor), file, NULL, G_FILE_MONITOR_EVENT_UNMOUNTED);
754           g_object_unref (file);
755         }
756       local_monitor->was_mounted = is_mounted;
757     }
758 }
759 #endif
760 
761 static void
g_local_file_monitor_start(GLocalFileMonitor * local_monitor,const gchar * filename,gboolean is_directory,GFileMonitorFlags flags,GMainContext * context)762 g_local_file_monitor_start (GLocalFileMonitor *local_monitor,
763                             const gchar       *filename,
764                             gboolean           is_directory,
765                             GFileMonitorFlags  flags,
766                             GMainContext      *context)
767 {
768   GLocalFileMonitorClass *class = G_LOCAL_FILE_MONITOR_GET_CLASS (local_monitor);
769   GFileMonitorSource *source;
770 
771   g_return_if_fail (G_IS_LOCAL_FILE_MONITOR (local_monitor));
772 
773   g_assert (!local_monitor->source);
774 
775   source = g_file_monitor_source_new (local_monitor, filename, is_directory, flags);
776   local_monitor->source = source; /* owns the ref */
777 
778   if (is_directory && !class->mount_notify && (flags & G_FILE_MONITOR_WATCH_MOUNTS))
779     {
780 #ifdef G_OS_WIN32
781       /*claim everything was mounted */
782       local_monitor->was_mounted = TRUE;
783 #else
784       GUnixMountEntry *mount;
785 
786       /* Emulate unmount detection */
787 
788       mount = g_unix_mount_at (local_monitor->source->dirname, NULL);
789 
790       local_monitor->was_mounted = mount != NULL;
791 
792       if (mount)
793         g_unix_mount_free (mount);
794 
795       local_monitor->mount_monitor = g_unix_mount_monitor_get ();
796       g_signal_connect_object (local_monitor->mount_monitor, "mounts-changed",
797                                G_CALLBACK (g_local_file_monitor_mounts_changed), local_monitor, 0);
798 #endif
799     }
800 
801   g_source_attach ((GSource *) source, context);
802 
803   G_LOCAL_FILE_MONITOR_GET_CLASS (local_monitor)->start (local_monitor,
804                                                          source->dirname, source->basename, source->filename,
805                                                          source);
806 }
807 
808 static void
g_local_file_monitor_dispose(GObject * object)809 g_local_file_monitor_dispose (GObject *object)
810 {
811   GLocalFileMonitor *local_monitor = G_LOCAL_FILE_MONITOR (object);
812 
813   g_file_monitor_source_dispose (local_monitor->source);
814 
815   G_OBJECT_CLASS (g_local_file_monitor_parent_class)->dispose (object);
816 }
817 
818 static void
g_local_file_monitor_finalize(GObject * object)819 g_local_file_monitor_finalize (GObject *object)
820 {
821   GLocalFileMonitor *local_monitor = G_LOCAL_FILE_MONITOR (object);
822 
823   g_source_unref ((GSource *) local_monitor->source);
824 
825   G_OBJECT_CLASS (g_local_file_monitor_parent_class)->finalize (object);
826 }
827 
828 static void
g_local_file_monitor_init(GLocalFileMonitor * local_monitor)829 g_local_file_monitor_init (GLocalFileMonitor* local_monitor)
830 {
831 }
832 
g_local_file_monitor_class_init(GLocalFileMonitorClass * class)833 static void g_local_file_monitor_class_init (GLocalFileMonitorClass *class)
834 {
835   GObjectClass *gobject_class = G_OBJECT_CLASS (class);
836 
837   gobject_class->get_property = g_local_file_monitor_get_property;
838   gobject_class->set_property = g_local_file_monitor_set_property;
839   gobject_class->dispose = g_local_file_monitor_dispose;
840   gobject_class->finalize = g_local_file_monitor_finalize;
841 
842   g_object_class_override_property (gobject_class, PROP_RATE_LIMIT, "rate-limit");
843 }
844 
845 static GLocalFileMonitor *
g_local_file_monitor_new(gboolean is_remote_fs,gboolean is_directory,GError ** error)846 g_local_file_monitor_new (gboolean   is_remote_fs,
847                           gboolean   is_directory,
848                           GError   **error)
849 {
850   GType type = G_TYPE_INVALID;
851 
852   if (is_remote_fs)
853     type = _g_io_module_get_default_type (G_NFS_FILE_MONITOR_EXTENSION_POINT_NAME,
854                                           "GIO_USE_FILE_MONITOR",
855                                           G_STRUCT_OFFSET (GLocalFileMonitorClass, is_supported));
856 
857   /* Fallback rather to poll file monitor for remote files, see gfile.c. */
858   if (type == G_TYPE_INVALID && (!is_remote_fs || is_directory))
859     type = _g_io_module_get_default_type (G_LOCAL_FILE_MONITOR_EXTENSION_POINT_NAME,
860                                           "GIO_USE_FILE_MONITOR",
861                                           G_STRUCT_OFFSET (GLocalFileMonitorClass, is_supported));
862 
863   if (type == G_TYPE_INVALID)
864     {
865       g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_FAILED,
866                            _("Unable to find default local file monitor type"));
867       return NULL;
868     }
869 
870   return g_object_new (type, NULL);
871 }
872 
873 GFileMonitor *
g_local_file_monitor_new_for_path(const gchar * pathname,gboolean is_directory,GFileMonitorFlags flags,GError ** error)874 g_local_file_monitor_new_for_path (const gchar        *pathname,
875                                    gboolean            is_directory,
876                                    GFileMonitorFlags   flags,
877                                    GError            **error)
878 {
879   GLocalFileMonitor *monitor;
880   gboolean is_remote_fs;
881 
882   is_remote_fs = g_local_file_is_nfs_home (pathname);
883 
884   monitor = g_local_file_monitor_new (is_remote_fs, is_directory, error);
885 
886   if (monitor)
887     g_local_file_monitor_start (monitor, pathname, is_directory, flags, g_main_context_get_thread_default ());
888 
889   return G_FILE_MONITOR (monitor);
890 }
891 
892 GFileMonitor *
g_local_file_monitor_new_in_worker(const gchar * pathname,gboolean is_directory,GFileMonitorFlags flags,GFileMonitorCallback callback,gpointer user_data,GClosureNotify destroy_user_data,GError ** error)893 g_local_file_monitor_new_in_worker (const gchar           *pathname,
894                                     gboolean               is_directory,
895                                     GFileMonitorFlags      flags,
896                                     GFileMonitorCallback   callback,
897                                     gpointer               user_data,
898                                     GClosureNotify         destroy_user_data,
899                                     GError               **error)
900 {
901   GLocalFileMonitor *monitor;
902   gboolean is_remote_fs;
903 
904   is_remote_fs = g_local_file_is_nfs_home (pathname);
905 
906   monitor = g_local_file_monitor_new (is_remote_fs, is_directory, error);
907 
908   if (monitor)
909     {
910       if (callback)
911         g_signal_connect_data (monitor, "changed", G_CALLBACK (callback),
912                                user_data, destroy_user_data, 0  /* flags */);
913 
914       g_local_file_monitor_start (monitor, pathname, is_directory, flags, GLIB_PRIVATE_CALL(g_get_worker_context) ());
915     }
916 
917   return G_FILE_MONITOR (monitor);
918 }
919