On 11.02.2018 14:30, Stef Bon wrote: > 2018-02-09 23:45 GMT+01:00 Kirill Tkhai <ktkhai@xxxxxxxxxxxxx>: >> On 09.02.2018 23:56, Andrew Morton wrote: >>> On Fri, 9 Feb 2018 18:04:54 +0300 Kirill Tkhai <ktkhai@xxxxxxxxxxxxx> wrote: >>> >>>> Watch descriptor is id of the watch created by inotify_add_watch(). >>>> It is allocated in inotify_add_to_idr(), and takes the numbers >>>> starting from 1. Every new inotify watch obtains next available >>>> number (usually, old + 1), as served by idr_alloc_cyclic(). >>>> >>>> CRIU (Checkpoint/Restore In Userspace) project supports inotify >>>> files, and restores watched descriptors with the same numbers, >>>> they had before dump. Since there was no kernel support, we >>>> had to use cycle to add a watch with specific descriptor id: >>>> >>>> while (1) { >>>> int wd; >>>> >>>> wd = inotify_add_watch(inotify_fd, path, mask); >>>> if (wd < 0) { >>>> break; >>>> } else if (wd == desired_wd_id) { >>>> ret = 0; >>>> break; >>>> } >>>> >>>> inotify_rm_watch(inotify_fd, wd); >>>> } >>>> >>>> (You may find the actual code at the below link: >>>> https://github.com/checkpoint-restore/criu/blob/v3.7/criu/fsnotify.c#L577) > > Well using a ioctl command to force a specific wd is possible, but > isn't it also possible > to do a "freeze" of all (inotify) watches which are involved, and > "unfreeze" when restoring? Regarding C/R, all inotifies are involved ;) Also, all regular files, sockets, memory mappings, etc. Checkpoint code attaches to a process via ptrace() and injects parasite code, which collects data and metadata of all the process's entities. It's rather difficult action, because several processes may be checkpointed, and they may share files/memory mappings/fs/etc. Also, they may be related to different namespaces. It's long to tell. You may dive into CRIU code, if you're interested. Then, restore code tries to recreate the processes from ground (possible, on another physical machine). It uses standard linux system calls to do that, i.e., it starts from clone() and then creates everything else. When there is the time to restore a file (inotify in our case), standard linux inotify_init1() is called. We create the inotify fd, then dup2() it to appropriate number. Then, we need to add watched files/directories to the inotify. And they must be added with the same watch descriptor id, as they was at checkpoint time. We use inotify_add_watch() and it returns id == 1, as you can see in kernel code. But we need another id, say, 0xfffff. And there is no syscall like dup2() for inotify watch descriptors. So, we use cyclic inotify_add_watch()/inotify_rm_watch() as next inotify_add_watch() returns incremented id (see the kernel) despite inotify_rm_watch() was called to remove old. After 0xfffff-1 iterations, inotify_add_watch() reaches id we need and returns it. This scheme is very slow, and the patch allows to restory inotify using 2 syscalls only (ioctl+inotify_add_watch). So, answering your question: No, it's not possible to use freeze/unfreeze to do that. Kirill