diff options
author | Andreas Politz <politza@hochschule-trier.de> | 2017-03-26 09:21:56 +0200 |
---|---|---|
committer | Michael Albinus <michael.albinus@gmx.de> | 2017-03-26 09:21:56 +0200 |
commit | 158bb8555dfefa50f6118be6794d0424cc52d291 (patch) | |
tree | e55be569b5898834d9a1e8b25586b1ddeb597587 /lisp/filenotify.el | |
parent | 9278d904af13c3c083defdcbf5fa21260d4457c3 (diff) | |
download | emacs-158bb8555dfefa50f6118be6794d0424cc52d291.tar.gz emacs-158bb8555dfefa50f6118be6794d0424cc52d291.tar.bz2 emacs-158bb8555dfefa50f6118be6794d0424cc52d291.zip |
Fix issues regarding inotify file-notification
Remove special code handling the inotify back-end.
* lisp/filenotify.el (file-notify--watch): New struct
representing a file-watch.
(file-notify-descriptors): Use the new struct as hash-value.
(file-notify-handle-event): Check that event is a cons.
(file-notify--rm-descriptor, file-notify--event-watched-file)
(file-notify--event-file-name, file-notify--event-file1-name)
(file-notify-callback, file-notify-add-watch)
(file-notify-rm-watch, file-notify-valid-p): Use new struct.
Remove special code handling inotify descriptors. Remove code
handling multiple clients per descriptor.
(file-notify--descriptor): Remove unused function.
Let inotify-add-watch return a unique descriptor on every
call, like every other back-end does (Bug#26126). Prevent
multiple clients from interfering with each other, when
watching a shared descriptor.
* src/inotify.c (watch_list): Extend the format by including a
id and the provided mask.
(INOTIFY_DEFAULT_MASK): Default mask used for all clients.
(make_watch_descriptor): Removed.
(make_lispy_mask, lispy_mask_match_p): New functions.
(inotifyevent_to_event): Match event against the mask provided
by the client.
(add_watch, remove_descriptor, remove_watch): New functions
for managing the watch_list.
(inotify_callback): Use the new functions.
(Finotify_add_watch, Finotify_rm_watch): Remove deprecated
flags from documentation. Add check for validity of provided
descriptor. Use the new functions. Use the default mask.
(INOTIFY_DEBUG): Add new debug conditional.
(inotify-watch-list, inotify-allocated-p): New debug functions.
(symbol_to_inotifymask, syms_of_inotify): Remove deprecated symbols.
* test/lisp/filenotify-tests.el:
(file-notify-test02-rm-watch): Remove expected failure for inotify.
Diffstat (limited to 'lisp/filenotify.el')
-rw-r--r-- | lisp/filenotify.el | 432 |
1 files changed, 198 insertions, 234 deletions
diff --git a/lisp/filenotify.el b/lisp/filenotify.el index 80e9f898b2e..0f8c945a79e 100644 --- a/lisp/filenotify.el +++ b/lisp/filenotify.el @@ -40,41 +40,42 @@ The value is the name of the low-level file notification package to be used for local file systems. Remote file notifications could use another implementation.") +(cl-defstruct (file-notify--watch + (:constructor nil) + (:constructor file-notify--watch-make (directory filename callback))) + ;; Watched directory + directory + ;; Watched relative filename, nil if watching the directory. + filename + ;; Function to propagate events to + callback) + +(defun file-notify--watch-absolute-filename (watch) + (if (file-notify--watch-filename watch) + (expand-file-name + (file-notify--watch-filename watch) + (file-notify--watch-directory watch)) + (file-notify--watch-directory watch))) + (defvar file-notify-descriptors (make-hash-table :test 'equal) "Hash table for registered file notification descriptors. A key in this hash table is the descriptor as returned from `inotify', `kqueue', `gfilenotify', `w32notify' or a file name -handler. The value in the hash table is a list - - (DIR (FILE . CALLBACK) (FILE . CALLBACK) ...) - -Several values for a given DIR happen only for `inotify', when -different files from the same directory are watched.") +handler. The value in the hash table is file-notify--watch +struct.") (defun file-notify--rm-descriptor (descriptor) "Remove DESCRIPTOR from `file-notify-descriptors'. -DESCRIPTOR should be an object returned by `file-notify-add-watch'. -If it is registered in `file-notify-descriptors', a stopped event is sent." - (let* ((desc (if (consp descriptor) (car descriptor) descriptor)) - (registered (gethash desc file-notify-descriptors)) - (file (if (consp descriptor) (cdr descriptor) (cl-caadr registered))) - (dir (car registered))) - - (when (consp registered) - ;; Send `stopped' event. - (when (consp (assoc file (cdr registered))) +DESCRIPTOR should be an object returned by +`file-notify-add-watch'. If it is registered in +`file-notify-descriptors', a stopped event is sent." + (when-let (watch (gethash descriptor file-notify-descriptors)) + ;; Send `stopped' event. + (unwind-protect (funcall - (cdr (assoc file (cdr registered))) - `(,descriptor stopped ,(if file (expand-file-name file dir) dir)))) - - ;; Modify `file-notify-descriptors'. - (if (not file) - (remhash desc file-notify-descriptors) - (setcdr registered - (delete (assoc file (cdr registered)) (cdr registered))) - (if (null (cdr registered)) - (remhash desc file-notify-descriptors) - (puthash desc registered file-notify-descriptors)))))) + (file-notify--watch-callback watch) + `(,descriptor stopped ,(file-notify--watch-absolute-filename watch))) + (remhash descriptor file-notify-descriptors)))) ;; This function is used by `inotify', `kqueue', `gfilenotify' and ;; `w32notify' events. @@ -88,7 +89,8 @@ If EVENT is a filewatch event, call its callback. It has the format Otherwise, signal a `file-notify-error'." (interactive "e") ;;(message "file-notify-handle-event %S" event) - (if (and (eq (car event) 'file-notify) + (if (and (consp event) + (eq (car event) 'file-notify) (>= (length event) 3)) (funcall (nth 2 event) (nth 1 event)) (signal 'file-notify-error @@ -96,33 +98,33 @@ Otherwise, signal a `file-notify-error'." ;; Needed for `inotify' and `w32notify'. In the latter case, COOKIE is nil. (defvar file-notify--pending-event nil - "A pending file notification events for a future `renamed' action. + "A pending file notification event for a future `renamed' action. It is a form ((DESCRIPTOR ACTION FILE [FILE1-OR-COOKIE]) CALLBACK).") (defun file-notify--event-watched-file (event) "Return file or directory being watched. Could be different from the directory watched by the backend library." - (let* ((desc (if (consp (car event)) (caar event) (car event))) - (registered (gethash desc file-notify-descriptors)) - (file (if (consp (car event)) (cdar event) (cl-caadr registered))) - (dir (car registered))) - (if file (expand-file-name file dir) dir))) + (when-let (watch (gethash (car event) file-notify-descriptors)) + (file-notify--watch-absolute-filename watch))) (defun file-notify--event-file-name (event) "Return file name of file notification event, or nil." - (directory-file-name - (expand-file-name - (or (and (stringp (nth 2 event)) (nth 2 event)) "") - (car (gethash (car event) file-notify-descriptors))))) + (when-let (watch (gethash (car event) file-notify-descriptors)) + (directory-file-name + (expand-file-name + (or (and (stringp (nth 2 event)) (nth 2 event)) "") + (file-notify--watch-directory watch))))) ;; Only `gfilenotify' could return two file names. (defun file-notify--event-file1-name (event) "Return second file name of file notification event, or nil. This is available in case a file has been moved." - (and (stringp (nth 3 event)) - (directory-file-name - (expand-file-name - (nth 3 event) (car (gethash (car event) file-notify-descriptors)))))) + (when-let (watch (gethash (car event) file-notify-descriptors)) + (and (stringp (nth 3 event)) + (directory-file-name + (expand-file-name + (nth 3 event) + (file-notify--watch-directory watch)))))) ;; Cookies are offered by `inotify' only. (defun file-notify--event-cookie (event) @@ -130,21 +132,6 @@ This is available in case a file has been moved." This is available in case a file has been moved." (nth 3 event)) -;; `inotify' returns the same descriptor when the file (directory) -;; uses the same inode. We want to distinguish, and apply a virtual -;; descriptor which make the difference. -(defun file-notify--descriptor (desc file) - "Return the descriptor to be used in `file-notify-*-watch'. -For `gfilenotify' and `w32notify' it is the same descriptor as -used in the low-level file notification package." - (if (and (natnump desc) (eq file-notify--library 'inotify)) - (cons desc - (and (stringp file) - (car (assoc - (file-name-nondirectory file) - (gethash desc file-notify-descriptors))))) - desc)) - ;; The callback function used to map between specific flags of the ;; respective file notifications, and the ones we return. (defun file-notify-callback (event) @@ -152,138 +139,125 @@ used in the low-level file notification package." EVENT is the cadr of the event in `file-notify-handle-event' \(DESCRIPTOR ACTIONS FILE [FILE1-OR-COOKIE])." (let* ((desc (car event)) - (registered (gethash desc file-notify-descriptors)) + (watch (gethash desc file-notify-descriptors)) (actions (nth 1 event)) (file (file-notify--event-file-name event)) - file1 callback pending-event stopped) + file1 pending-event stopped) ;; Make actions a list. (unless (consp actions) (setq actions (cons actions nil))) - ;; Loop over registered entries. In fact, more than one entry - ;; happens only for `inotify'. - (dolist (entry (cdr registered)) - - ;; Check, that event is meant for us. - (unless (setq callback (cdr entry)) - (setq actions nil)) - + (when watch ;; Loop over actions. In fact, more than one action happens only ;; for `inotify' and `kqueue'. - (dolist (action actions) - - ;; Send pending event, if it doesn't match. - (when (and file-notify--pending-event - ;; The cookie doesn't match. - (not (eq (file-notify--event-cookie - (car file-notify--pending-event)) - (file-notify--event-cookie event))) - (or - ;; inotify. - (and (eq (nth 1 (car file-notify--pending-event)) - 'moved-from) - (not (eq action 'moved-to))) - ;; w32notify. - (and (eq (nth 1 (car file-notify--pending-event)) - 'renamed-from) - (not (eq action 'renamed-to))))) - (setq pending-event file-notify--pending-event - file-notify--pending-event nil) - (setcar (cdar pending-event) 'deleted)) - - ;; Map action. We ignore all events which cannot be mapped. - (setq action - (cond - ((memq action - '(attribute-changed changed created deleted renamed)) - action) - ((memq action '(moved rename)) - ;; The kqueue rename event does not return file1 in - ;; case a file monitor is established. - (if (setq file1 (file-notify--event-file1-name event)) - 'renamed 'deleted)) - ((eq action 'ignored) - (setq stopped t actions nil)) - ((memq action '(attrib link)) 'attribute-changed) - ((memq action '(create added)) 'created) - ((memq action '(modify modified write)) 'changed) - ((memq action '(delete delete-self move-self removed)) 'deleted) - ;; Make the event pending. - ((memq action '(moved-from renamed-from)) - (setq file-notify--pending-event - `((,desc ,action ,file ,(file-notify--event-cookie event)) - ,callback)) - nil) - ;; Look for pending event. - ((memq action '(moved-to renamed-to)) - (if (null file-notify--pending-event) - 'created - (setq file1 file - file (file-notify--event-file-name - (car file-notify--pending-event))) - ;; If the source is handled by another watch, we - ;; must fire the rename event there as well. - (when (not (equal (file-notify--descriptor desc file1) - (file-notify--descriptor - (caar file-notify--pending-event) - (file-notify--event-file-name - file-notify--pending-event)))) - (setq pending-event - `((,(caar file-notify--pending-event) - renamed ,file ,file1) - ,(cadr file-notify--pending-event)))) - (setq file-notify--pending-event nil) - 'renamed)))) - - ;; Apply pending callback. - (when pending-event - (setcar - (car pending-event) - (file-notify--descriptor - (caar pending-event) - (file-notify--event-file-name file-notify--pending-event))) - (funcall (cadr pending-event) (car pending-event)) - (setq pending-event nil)) - - ;; Apply callback. - (when (and action - (or - ;; If there is no relative file name for that watch, - ;; we watch the whole directory. - (null (nth 0 entry)) - ;; File matches. - (string-equal - (nth 0 entry) (file-name-nondirectory file)) - ;; Directory matches. - (string-equal - (file-name-nondirectory file) - (file-name-nondirectory (car registered))) - ;; File1 matches. - (and (stringp file1) - (string-equal - (nth 0 entry) (file-name-nondirectory file1))))) - ;;(message - ;;"file-notify-callback %S %S %S %S %S" - ;;(file-notify--descriptor desc (car entry)) - ;;action file file1 registered) - (if file1 - (funcall - callback - `(,(file-notify--descriptor desc (car entry)) - ,action ,file ,file1)) - (funcall - callback - `(,(file-notify--descriptor desc (car entry)) ,action ,file)))) - - ;; Send `stopped' event. - (when (or stopped - (and (memq action '(deleted renamed)) - ;; Not, when a file is backed up. - (not (and (stringp file1) (backup-file-name-p file1))) - ;; Watched file or directory is concerned. - (string-equal - file (file-notify--event-watched-file event)))) - (file-notify-rm-watch (file-notify--descriptor desc (car entry)))))))) + (while actions + (let ((action (pop actions))) + ;; Send pending event, if it doesn't match. + (when (and file-notify--pending-event + ;; The cookie doesn't match. + (not (eq (file-notify--event-cookie + (car file-notify--pending-event)) + (file-notify--event-cookie event))) + (or + ;; inotify. + (and (eq (nth 1 (car file-notify--pending-event)) + 'moved-from) + (not (eq action 'moved-to))) + ;; w32notify. + (and (eq (nth 1 (car file-notify--pending-event)) + 'renamed-from) + (not (eq action 'renamed-to))))) + (setq pending-event file-notify--pending-event + file-notify--pending-event nil) + (setcar (cdar pending-event) 'deleted)) + + ;; Map action. We ignore all events which cannot be mapped. + (setq action + (cond + ((memq action + '(attribute-changed changed created deleted renamed)) + action) + ((memq action '(moved rename)) + ;; The kqueue rename event does not return file1 in + ;; case a file monitor is established. + (if (setq file1 (file-notify--event-file1-name event)) + 'renamed 'deleted)) + ((eq action 'ignored) + (setq stopped t actions nil)) + ((memq action '(attrib link)) 'attribute-changed) + ((memq action '(create added)) 'created) + ((memq action '(modify modified write)) 'changed) + ((memq action '(delete delete-self move-self removed)) 'deleted) + ;; Make the event pending. + ((memq action '(moved-from renamed-from)) + (setq file-notify--pending-event + `((,desc ,action ,file ,(file-notify--event-cookie event)) + ,(file-notify--watch-callback watch))) + nil) + ;; Look for pending event. + ((memq action '(moved-to renamed-to)) + (if (null file-notify--pending-event) + 'created + (setq file1 file + file (file-notify--event-file-name + (car file-notify--pending-event))) + ;; If the source is handled by another watch, we + ;; must fire the rename event there as well. + (when (not (equal desc (caar file-notify--pending-event))) + (setq pending-event + `((,(caar file-notify--pending-event) + renamed ,file ,file1) + ,(cadr file-notify--pending-event)))) + (setq file-notify--pending-event nil) + 'renamed)))) + + ;; Apply pending callback. + (when pending-event + (setcar + (car pending-event) + (caar pending-event)) + (funcall (cadr pending-event) (car pending-event)) + (setq pending-event nil)) + + ;; Apply callback. + (when (and action + (or + ;; If there is no relative file name for that watch, + ;; we watch the whole directory. + (null (file-notify--watch-filename watch)) + ;; File matches. + (string-equal + (file-notify--watch-filename watch) + (file-name-nondirectory file)) + ;; Directory matches. + (string-equal + (file-name-nondirectory file) + (file-name-nondirectory + (file-notify--watch-directory watch))) + ;; File1 matches. + (and (stringp file1) + (string-equal + (file-notify--watch-filename watch) + (file-name-nondirectory file1))))) + ;;(message + ;;"file-notify-callback %S %S %S %S %S" + ;;desc + ;;action file file1 watch) + (if file1 + (funcall (file-notify--watch-callback watch) + `(,desc ,action ,file ,file1)) + (funcall (file-notify--watch-callback watch) + `(,desc ,action ,file)))) + + ;; Send `stopped' event. + (when (or stopped + (and (memq action '(deleted renamed)) + ;; Not, when a file is backed up. + (not (and (stringp file1) (backup-file-name-p file1))) + ;; Watched file or directory is concerned. + (string-equal + file (file-notify--event-watched-file event)))) + (file-notify-rm-watch desc))))))) ;; `kqueue', `gfilenotify' and `w32notify' return a unique descriptor ;; for every `file-notify-add-watch', while `inotify' returns a unique @@ -339,7 +313,7 @@ FILE is the name of the file whose event is being reported." (if (file-directory-p file) file (file-name-directory file)))) - desc func l-flags registered entry) + desc func l-flags) (unless (file-directory-p dir) (signal 'file-notify-error `("Directory does not exist" ,dir))) @@ -391,66 +365,46 @@ FILE is the name of the file whose event is being reported." l-flags 'file-notify-callback))) ;; Modify `file-notify-descriptors'. - (setq file (unless (file-directory-p file) (file-name-nondirectory file)) - desc (if (consp desc) (car desc) desc) - registered (gethash desc file-notify-descriptors) - entry `(,file . ,callback)) - (unless (member entry (cdr registered)) - (puthash desc `(,dir ,entry . ,(cdr registered)) file-notify-descriptors)) - + (let ((watch (file-notify--watch-make + dir + (unless (file-directory-p file) (file-name-nondirectory file)) + callback))) + (puthash desc watch file-notify-descriptors)) ;; Return descriptor. - (file-notify--descriptor desc file))) + desc)) (defun file-notify-rm-watch (descriptor) "Remove an existing watch specified by its DESCRIPTOR. DESCRIPTOR should be an object returned by `file-notify-add-watch'." - (let* ((desc (if (consp descriptor) (car descriptor) descriptor)) - (file (if (consp descriptor) (cdr descriptor))) - (registered (gethash desc file-notify-descriptors)) - (dir (car registered)) - (handler (and (stringp dir) - (find-file-name-handler dir 'file-notify-rm-watch)))) - - (when (stringp dir) - ;; Call low-level function. - (when (or (not file) - (and (= (length (cdr registered)) 1) - (assoc file (cdr registered)))) - (condition-case nil - (if handler - ;; A file name handler could exist even if there is no local - ;; file notification support. - (funcall handler 'file-notify-rm-watch descriptor) - - (funcall - (cond - ((eq file-notify--library 'inotify) 'inotify-rm-watch) - ((eq file-notify--library 'kqueue) 'kqueue-rm-watch) - ((eq file-notify--library 'gfilenotify) 'gfile-rm-watch) - ((eq file-notify--library 'w32notify) 'w32notify-rm-watch)) - desc)) - (file-notify-error nil))) - - ;; Modify `file-notify-descriptors'. - (file-notify--rm-descriptor descriptor)))) + (when-let (watch (gethash descriptor file-notify-descriptors)) + (let ((handler (find-file-name-handler + (file-notify--watch-directory watch) + 'file-notify-rm-watch))) + (condition-case nil + (if handler + ;; A file name handler could exist even if there is no local + ;; file notification support. + (funcall handler 'file-notify-rm-watch descriptor) + + (funcall + (cond + ((eq file-notify--library 'inotify) 'inotify-rm-watch) + ((eq file-notify--library 'kqueue) 'kqueue-rm-watch) + ((eq file-notify--library 'gfilenotify) 'gfile-rm-watch) + ((eq file-notify--library 'w32notify) 'w32notify-rm-watch)) + descriptor)) + (file-notify-error nil))) + ;; Modify `file-notify-descriptors'. + (file-notify--rm-descriptor descriptor))) (defun file-notify-valid-p (descriptor) "Check a watch specified by its DESCRIPTOR. DESCRIPTOR should be an object returned by `file-notify-add-watch'." - (let* ((desc (if (consp descriptor) (car descriptor) descriptor)) - (file (if (consp descriptor) (cdr descriptor))) - (registered (gethash desc file-notify-descriptors)) - (dir (car registered)) - handler) - - (when (stringp dir) - (setq handler (find-file-name-handler dir 'file-notify-valid-p)) - - (and (or ;; It is a directory. - (not file) - ;; The file is registered. - (assoc file (cdr registered))) - (if handler + (when-let (watch (gethash descriptor file-notify-descriptors)) + (let ((handler (find-file-name-handler + (file-notify--watch-directory watch) + 'file-notify-valid-p))) + (and (if handler ;; A file name handler could exist even if there is no ;; local file notification support. (funcall handler 'file-notify-valid-p descriptor) @@ -460,9 +414,19 @@ DESCRIPTOR should be an object returned by `file-notify-add-watch'." ((eq file-notify--library 'kqueue) 'kqueue-valid-p) ((eq file-notify--library 'gfilenotify) 'gfile-valid-p) ((eq file-notify--library 'w32notify) 'w32notify-valid-p)) - desc)) + descriptor)) t)))) + +;; TODO: +;; * Watching a /dir/file may receive events for dir. +;; (This may be the desired behaviour.) +;; * Watching a file in a already watched directory +;; If the file is created and *then* a watch is added to that file, the +;; watch might receive events which occured prior to it being created, +;; due to the way events are propagated during idle time. Note: This +;; may be perfectly acceptable. + ;; The end: (provide 'filenotify) |