1 /*******************************************************************************
2 Copyright (c) 2011, 2012 Dmitry Matveev <me@dmitrymatveev.co.uk>
4 Permission is hereby granted, free of charge, to any person obtaining a copy
5 of this software and associated documentation files (the "Software"), to deal
6 in the Software without restriction, including without limitation the rights
7 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8 copies of the Software, and to permit persons to whom the Software is
9 furnished to do so, subject to the following conditions:
11 The above copyright notice and this permission notice shall be included in
12 all copies or substantial portions of the Software.
14 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
19 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21 *******************************************************************************/
24 #include <sys/types.h>
25 #include <sys/event.h>
27 #include <sys/socket.h>
28 #include <gio/glocalfile.h>
29 #include <gio/glocalfilemonitor.h>
30 #include <gio/gfile.h>
36 #include "kqueue-helper.h"
37 #include "kqueue-utils.h"
38 #include "kqueue-thread.h"
39 #include "kqueue-missing.h"
40 #include "kqueue-exclusions.h"
42 static gboolean kh_debug_enabled
= FALSE
;
43 #define KH_W if (kh_debug_enabled) g_warning
45 static GHashTable
*subs_hash_table
= NULL
;
46 G_LOCK_DEFINE_STATIC (hash_lock
);
48 static int kqueue_descriptor
= -1;
49 static int kqueue_socket_pair
[] = {-1, -1};
50 static pthread_t kqueue_thread
;
53 void _kh_file_appeared_cb (kqueue_sub
*sub
);
56 * accessor function for kqueue_descriptor
59 get_kqueue_descriptor()
61 return kqueue_descriptor
;
65 * convert_kqueue_events_to_gio:
66 * @flags: a set of kqueue filter flags
67 * @done: a pointer to #gboolean indicating that the
68 * conversion has been done (out)
70 * Translates kqueue filter flags into GIO event flags.
72 * Returns: a #GFileMonitorEvent
74 static GFileMonitorEvent
75 convert_kqueue_events_to_gio (uint32_t flags
, gboolean
*done
)
77 g_assert (done
!= NULL
);
80 /* TODO: The following notifications should be emulated, if possible:
81 * - G_FILE_MONITOR_EVENT_PRE_UNMOUNT
83 if (flags
& NOTE_DELETE
)
86 return G_FILE_MONITOR_EVENT_DELETED
;
88 if (flags
& NOTE_ATTRIB
)
91 return G_FILE_MONITOR_EVENT_ATTRIBUTE_CHANGED
;
93 if (flags
& (NOTE_WRITE
| NOTE_EXTEND
))
96 return G_FILE_MONITOR_EVENT_CHANGED
;
98 if (flags
& NOTE_RENAME
)
101 return G_FILE_MONITOR_EVENT_MOVED
;
103 if (flags
& NOTE_REVOKE
)
106 return G_FILE_MONITOR_EVENT_UNMOUNTED
;
115 GFileMonitorSource
*source
;
120 * @udata: a pointer to user data (#handle_context).
121 * @path: file name of a new file.
122 * @inode: inode number of a new file.
124 * A callback function for the directory diff calculation routine,
125 * produces G_FILE_MONITOR_EVENT_CREATED event for a created file.
128 handle_created (void *udata
, const char *path
, ino_t inode
)
130 handle_ctx
*ctx
= NULL
;
133 ctx
= (handle_ctx
*) udata
;
134 g_assert (udata
!= NULL
);
135 g_assert (ctx
->sub
!= NULL
);
136 g_assert (ctx
->source
!= NULL
);
138 g_file_monitor_source_handle_event (ctx
->source
, G_FILE_MONITOR_EVENT_CREATED
, path
,
139 NULL
, NULL
, g_get_monotonic_time ());
144 * @udata: a pointer to user data (#handle_context).
145 * @path: file name of the removed file.
146 * @inode: inode number of the removed file.
148 * A callback function for the directory diff calculation routine,
149 * produces G_FILE_MONITOR_EVENT_DELETED event for a deleted file.
152 handle_deleted (void *udata
, const char *path
, ino_t inode
)
154 handle_ctx
*ctx
= NULL
;
157 ctx
= (handle_ctx
*) udata
;
158 g_assert (udata
!= NULL
);
159 g_assert (ctx
->sub
!= NULL
);
160 g_assert (ctx
->source
!= NULL
);
162 g_file_monitor_source_handle_event (ctx
->source
, G_FILE_MONITOR_EVENT_DELETED
, path
,
163 NULL
, NULL
, g_get_monotonic_time ());
168 * @udata: a pointer to user data (#handle_context).
169 * @from_path: file name of the source file.
170 * @from_inode: inode number of the source file.
171 * @to_path: file name of the replaced file.
172 * @to_inode: inode number of the replaced file.
174 * A callback function for the directory diff calculation routine,
175 * produces G_FILE_MONITOR_EVENT_RENAMED event on a move.
178 handle_moved (void *udata
,
179 const char *from_path
,
184 handle_ctx
*ctx
= NULL
;
189 ctx
= (handle_ctx
*) udata
;
190 g_assert (udata
!= NULL
);
191 g_assert (ctx
->sub
!= NULL
);
192 g_assert (ctx
->source
!= NULL
);
194 g_file_monitor_source_handle_event (ctx
->source
, G_FILE_MONITOR_EVENT_RENAMED
,
195 from_path
, to_path
, NULL
, g_get_monotonic_time ());
199 * handle_overwritten:
200 * @data: a pointer to user data (#handle_context).
201 * @path: file name of the overwritten file.
202 * @node: inode number of the overwritten file.
204 * A callback function for the directory diff calculation routine,
205 * produces G_FILE_MONITOR_EVENT_DELETED/CREATED event pair when
206 * an overwrite occurs in the directory (see dep-list for details).
209 handle_overwritten (void *udata
, const char *path
, ino_t inode
)
211 handle_ctx
*ctx
= NULL
;
214 ctx
= (handle_ctx
*) udata
;
215 g_assert (udata
!= NULL
);
216 g_assert (ctx
->sub
!= NULL
);
217 g_assert (ctx
->source
!= NULL
);
219 g_file_monitor_source_handle_event (ctx
->source
, G_FILE_MONITOR_EVENT_DELETED
,
220 path
, NULL
, NULL
, g_get_monotonic_time ());
222 g_file_monitor_source_handle_event (ctx
->source
, G_FILE_MONITOR_EVENT_CREATED
,
223 path
, NULL
, NULL
, g_get_monotonic_time ());
226 static const traverse_cbs cbs
= {
232 NULL
, /* many added */
233 NULL
, /* many removed */
234 NULL
, /* names updated */
239 _kh_dir_diff (kqueue_sub
*sub
, GFileMonitorSource
*source
)
244 g_assert (sub
!= NULL
);
245 g_assert (source
!= NULL
);
247 memset (&ctx
, 0, sizeof (handle_ctx
));
252 sub
->deps
= dl_listing (sub
->filename
);
254 dl_calculate (was
, sub
->deps
, &cbs
, &ctx
);
261 * process_kqueue_notifications:
266 * Processes notifications, coming from the kqueue thread.
268 * Reads notifications from the command file descriptor, emits the
269 * "changed" event on the appropriate monitor.
271 * A typical GIO Channel callback function.
276 process_kqueue_notifications (GIOChannel
*gioc
,
280 struct kqueue_notification n
;
281 kqueue_sub
*sub
= NULL
;
282 GFileMonitorSource
*source
= NULL
;
283 GFileMonitorEvent mask
= 0;
285 g_assert (kqueue_socket_pair
[0] != -1);
286 if (!_ku_read (kqueue_socket_pair
[0], &n
, sizeof (struct kqueue_notification
)))
288 KH_W ("Failed to read a kqueue notification, error %d", errno
);
293 sub
= (kqueue_sub
*) g_hash_table_lookup (subs_hash_table
, GINT_TO_POINTER (n
.fd
));
294 G_UNLOCK (hash_lock
);
298 KH_W ("Got a notification for a deleted or non-existing subscription %d",
303 source
= sub
->user_data
;
304 g_assert (source
!= NULL
);
306 if (n
.flags
& (NOTE_DELETE
| NOTE_REVOKE
))
313 _km_add_missing (sub
);
315 if (!(n
.flags
& NOTE_REVOKE
))
317 /* Note that NOTE_REVOKE is issued by the kqueue thread
318 * on EV_ERROR kevent. In this case, a file descriptor is
319 * already closed from the kqueue thread, no need to close
321 _kh_cancel_sub (sub
);
325 if (sub
->is_dir
&& n
.flags
& (NOTE_WRITE
| NOTE_EXTEND
))
327 _kh_dir_diff (sub
, source
);
328 n
.flags
&= ~(NOTE_WRITE
| NOTE_EXTEND
);
333 gboolean done
= FALSE
;
334 mask
= convert_kqueue_events_to_gio (n
.flags
, &done
);
336 g_file_monitor_source_handle_event (source
, mask
, NULL
, NULL
, NULL
, g_get_monotonic_time ());
347 * Kqueue backend startup code. Should be called only once.
349 * Returns: %TRUE on success, %FALSE otherwise.
352 _kh_startup_impl (gpointer unused
)
354 GIOChannel
*channel
= NULL
;
355 gboolean result
= FALSE
;
357 kqueue_descriptor
= kqueue ();
358 result
= (kqueue_descriptor
!= -1);
361 KH_W ("Failed to initialize kqueue\n!");
362 return GINT_TO_POINTER (FALSE
);
365 result
= socketpair (AF_UNIX
, SOCK_STREAM
, 0, kqueue_socket_pair
);
368 KH_W ("Failed to create socket pair\n!");
369 return GINT_TO_POINTER (FALSE
) ;
372 result
= pthread_create (&kqueue_thread
,
375 &kqueue_socket_pair
[1]);
378 KH_W ("Failed to run kqueue thread\n!");
379 return GINT_TO_POINTER (FALSE
);
382 _km_init (_kh_file_appeared_cb
);
384 channel
= g_io_channel_unix_new (kqueue_socket_pair
[0]);
385 g_io_add_watch (channel
, G_IO_IN
, process_kqueue_notifications
, NULL
);
387 subs_hash_table
= g_hash_table_new (g_direct_hash
, g_direct_equal
);
389 KH_W ("started gio kqueue backend\n");
390 return GINT_TO_POINTER (TRUE
);
396 * Kqueue backend initialization.
398 * Returns: %TRUE on success, %FALSE otherwise.
403 static GOnce init_once
= G_ONCE_INIT
;
404 g_once (&init_once
, _kh_startup_impl
, NULL
);
405 return GPOINTER_TO_INT (init_once
.retval
);
410 * _kh_start_watching:
411 * @sub: a #kqueue_sub
413 * Starts watching on a subscription.
415 * Returns: %TRUE on success, %FALSE otherwise.
418 _kh_start_watching (kqueue_sub
*sub
)
420 g_assert (kqueue_socket_pair
[0] != -1);
421 g_assert (sub
!= NULL
);
422 g_assert (sub
->filename
!= NULL
);
424 /* kqueue requires a file descriptor to monitor. Sad but true */
425 #if defined (O_EVTONLY)
426 sub
->fd
= open (sub
->filename
, O_EVTONLY
);
428 sub
->fd
= open (sub
->filename
, O_RDONLY
);
433 KH_W ("failed to open file %s (error %d)", sub
->filename
, errno
);
437 _ku_file_information (sub
->fd
, &sub
->is_dir
, NULL
);
440 /* I know, it is very bad to make such decisions in this way and here.
441 * We already do have an user_data at the #kqueue_sub, and it may point to
442 * GKqueueFileMonitor or GKqueueDirectoryMonitor. For a directory case,
443 * we need to scan in contents for the further diffs. Ideally this process
444 * should be delegated to the GKqueueDirectoryMonitor, but for now I will
445 * do it in a dirty way right here. */
449 sub
->deps
= dl_listing (sub
->filename
);
453 g_hash_table_insert (subs_hash_table
, GINT_TO_POINTER (sub
->fd
), sub
);
454 G_UNLOCK (hash_lock
);
456 _kqueue_thread_push_fd (sub
->fd
);
458 /* Bump the kqueue thread. It will pick up a new sub entry to monitor */
459 if (!_ku_write (kqueue_socket_pair
[0], "A", 1))
460 KH_W ("Failed to bump the kqueue thread (add fd, error %d)", errno
);
467 * @sub: a #kqueue_sub
469 * Adds a subscription for monitoring.
471 * This funciton tries to start watching a subscription with
472 * _kh_start_watching(). On failure, i.e. when a file does not exist yet,
473 * the subscription will be added to a list of missing files to continue
474 * watching when the file will appear.
479 _kh_add_sub (kqueue_sub
*sub
)
481 g_assert (sub
!= NULL
);
483 if (!_kh_start_watching (sub
))
484 _km_add_missing (sub
);
494 * Stops monitoring on a subscription.
499 _kh_cancel_sub (kqueue_sub
*sub
)
501 gboolean removed
= FALSE
;
502 g_assert (kqueue_socket_pair
[0] != -1);
503 g_assert (sub
!= NULL
);
508 removed
= g_hash_table_remove (subs_hash_table
, GINT_TO_POINTER (sub
->fd
));
509 G_UNLOCK (hash_lock
);
513 /* fd will be closed in the kqueue thread */
514 _kqueue_thread_remove_fd (sub
->fd
);
516 /* Bump the kqueue thread. It will pick up a new sub entry to remove*/
517 if (!_ku_write (kqueue_socket_pair
[0], "R", 1))
518 KH_W ("Failed to bump the kqueue thread (remove fd, error %d)", errno
);
526 * _kh_file_appeared_cb:
527 * @sub: a #kqueue_sub
529 * A callback function for kqueue-missing subsystem.
531 * Signals that a missing file has finally appeared in the filesystem.
532 * Emits %G_FILE_MONITOR_EVENT_CREATED.
535 _kh_file_appeared_cb (kqueue_sub
*sub
)
539 g_assert (sub
!= NULL
);
540 g_assert (sub
->filename
);
542 if (!g_file_test (sub
->filename
, G_FILE_TEST_EXISTS
))
545 child
= g_file_new_for_path (sub
->filename
);
547 g_file_monitor_emit_event (G_FILE_MONITOR (sub
->user_data
),
550 G_FILE_MONITOR_EVENT_CREATED
);
552 g_object_unref (child
);