Nadav, On Thu, Feb 10, 2022 at 10:42:43AM -0800, Nadav Amit wrote: > I think that the MADV_DONTNEED/PF-resolution “race" only affects usage-models > that handle the page-fault concurrently with UFFD-monitoring (using multiple > monitor threads or IO-uring as I try to do). At least for use-cases such as > live-migration. > > I think the scenario you have in mind is the following, which is resolved > with mmap_changing that Mike introduced some time ago: > > UFFD monitor App thread #0 App thread #1 > ------------ ------------- ------------- > #PF > UFFD Read > [#PF] > MADV_DONTNEED > mmap_changing = 1 > > userfaultfd_event_wait_completion() > [queue event, wait] > UFFD-copy > -EAGAIN since mmmap_changing > 0 > > > mmap_changing will keep being elevated, and UFFD-copy not served (fail) until > the monitor reads the UFFD event. The monitor, in this scenario, is single > threaded and therefore orders UFFD-read and UFFD-copy, preventing them from > racing. > > Assuming the monitor is smart enough to reevaluate the course of action after > MADV_DONTNEED is handled, it should be safe. Personally, I do not like the > burden that this scheme puts on the monitor, the fact it needs to retry or > even the return value [I think it should be EBUSY since immediate retry would > fail. With IO-uring, EAGAIN triggers an immediate retry, which is useless.] > Yet, concurrent UFFD-event/#PF can be handled properly by a smart monitor. > > *However*, userfaultfd events seem as very hard to use (to say the least) in > the following cases: > > 1. The UFFD-copy is issued by one thread and the UFFD-read is performed by > another. For me this is the most painful even if you may consider it > as “unorthodox”. It is very useful for performance, especially if the > UFFD-copy is large. This is definitely a valid use case for uffd, and IMHO that's a good base model when the uffd app is performance critical. > > 2. If the race is between 2 userfaultfd *events*. The events might not be > properly ordered (i.e., the order in which they are read does not reflect > the order in which they occurred) despite the use of *_userfaultfd_prep(), > since they are only queued (to be reported and trigger wake) by > userfaultfd_event_wait_completion(), after the VMA and PTEs were updated > and more importantly after mmap-lock was dropped. > > This means that if you have fork and MADV_DONTNEED, the monitor might see > their order inverted, and won’t be able to know whether the child has the > pages zapped or not. > > Other races are possible too, for instance between mremap() and munmap(). > In most cases the monitor might be able (with quite some work) to > figure out that the order of the events it received does not make sense > and the events must have been reordered. Yet, implementing something like > that is far from trivial and there are some cases that are probably > impossible to resolve just based on the UFFD read events. > > I personally addressed this issue with seccomp+ptrace to trap on > entry/exit to relevant syscalls (e.g., munmap, mmap, fork), and > prevent concurrent calls to obtain correct order of the events. It is far > from trivial and introduces some overheads, but I did not find a better > solution. Thanks for explaining. I also digged out the discussion threads between you and Mike and that's a good one too summarizing the problems: https://lore.kernel.org/all/5921BA80-F263-4F8D-B7E6-316CEB602B51@xxxxxxxxx/ Scenario 4 is kind of special imho along all those, because that's the only one that can be workarounded by user application by only copying pages one by one. I know you were even leveraging iouring in your local tree, so that's probably not a solution at all for you. But I'm just trying to start thinking without that scenario for now. Per my understanding, a major issue regarding the rest of the scenarios is ordering of uffd messages may not match with how things are happening. This actually contains two problems. First of all, mmap_sem is mostly held read for all page faults and most of the mm changes except e.g. fork, then we can never serialize them. Not to mention uffd events releases mmap_sem within prep and completion. Let's call it problem 1. The other problem 2 is we can never serialize faults against events. For problem 1, I do sense something that mmap_sem is just not suitable for uffd scenario. Say, we grant concurrent with most of the events like dontneed and mremap, but when uffd ordering is a concern we may not want to grant that concurrency. I'm wondering whether it means uffd may need its own semaphore to achieve this. So for all events that uffd cares we take write lock on a new uffd_sem after mmap_sem, meanwhile we don't release that uffd_sem after prep of events, not until completion (the message is read). It'll slow down uffd tracked systems but guarantees ordering. At the meantime, I'm wildly thinking whether we can tackle with the other problem by merging the page fault queue with the event queue, aka, event_wqh and fault_pending_wqh. Obviously we'll need to identify the messages when read() and conditionally move then into fault_wqh only if they come from page faults, but that seems doable? Not sure above makes any sense, as I could have missed something. Meanwhile I think even if we order all the messages to match with facts there're still some other issues that are outliers of this, but let's see how it sounds so far. Thanks, -- Peter Xu