mm: allow full handling of COW faults in ->fault handlers
authorJan Kara <jack@suse.cz>
Wed, 14 Dec 2016 23:07:18 +0000 (15:07 -0800)
committerLinus Torvalds <torvalds@linux-foundation.org>
Thu, 15 Dec 2016 00:04:09 +0000 (16:04 -0800)
Patch series "dax: Clear dirty bits after flushing caches", v5.

Patchset to clear dirty bits from radix tree of DAX inodes when caches
for corresponding pfns have been flushed.  In principle, these patches
enable handlers to easily update PTEs and do other work necessary to
finish the fault without duplicating the functionality present in the
generic code.  I'd like to thank Kirill and Ross for reviews of the
series!

This patch (of 20):

To allow full handling of COW faults add memcg field to struct vm_fault
and a return value of ->fault() handler meaning that COW fault is fully
handled and memcg charge must not be canceled.  This will allow us to
remove knowledge about special DAX locking from the generic fault code.

Link: http://lkml.kernel.org/r/1479460644-25076-9-git-send-email-jack@suse.cz
Signed-off-by: Jan Kara <jack@suse.cz>
Reviewed-by: Ross Zwisler <ross.zwisler@linux.intel.com>
Acked-by: Kirill A. Shutemov <kirill.shutemov@linux.intel.com>
Cc: Dan Williams <dan.j.williams@intel.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
include/linux/mm.h
mm/memory.c

index 39c17a2efcea085a76a1296ec1ff1814fd680bc5..6e25f4916d6fffcd148562efa4ca9606736d4009 100644 (file)
@@ -301,7 +301,8 @@ struct vm_fault {
                                         * the 'address' */
        pte_t orig_pte;                 /* Value of PTE at the time of fault */
 
-       struct page *cow_page;          /* Handler may choose to COW */
+       struct page *cow_page;          /* Page handler may use for COW fault */
+       struct mem_cgroup *memcg;       /* Cgroup cow_page belongs to */
        struct page *page;              /* ->fault handlers should return a
                                         * page here, unless VM_FAULT_NOPAGE
                                         * is set (which is also implied by
@@ -1103,6 +1104,7 @@ static inline void clear_page_pfmemalloc(struct page *page)
 #define VM_FAULT_RETRY 0x0400  /* ->fault blocked, must retry */
 #define VM_FAULT_FALLBACK 0x0800       /* huge page fault failed, fall back to small */
 #define VM_FAULT_DAX_LOCKED 0x1000     /* ->fault has locked DAX entry */
+#define VM_FAULT_DONE_COW   0x2000     /* ->fault has fully handled COW */
 
 #define VM_FAULT_HWPOISON_LARGE_MASK 0xf000 /* encodes hpage index for large hwpoison */
 
index cf74f7ca911b23b8f8525ce1c9230a89c7fbf822..02504cd4ca0e243cb63aaa99508932e48bd86799 100644 (file)
@@ -2844,9 +2844,8 @@ static int __do_fault(struct vm_fault *vmf)
        int ret;
 
        ret = vma->vm_ops->fault(vma, vmf);
-       if (unlikely(ret & (VM_FAULT_ERROR | VM_FAULT_NOPAGE | VM_FAULT_RETRY)))
-               return ret;
-       if (ret & VM_FAULT_DAX_LOCKED)
+       if (unlikely(ret & (VM_FAULT_ERROR | VM_FAULT_NOPAGE | VM_FAULT_RETRY |
+                           VM_FAULT_DAX_LOCKED | VM_FAULT_DONE_COW)))
                return ret;
 
        if (unlikely(PageHWPoison(vmf->page))) {
@@ -3226,7 +3225,6 @@ static int do_read_fault(struct vm_fault *vmf)
 static int do_cow_fault(struct vm_fault *vmf)
 {
        struct vm_area_struct *vma = vmf->vma;
-       struct mem_cgroup *memcg;
        int ret;
 
        if (unlikely(anon_vma_prepare(vma)))
@@ -3237,7 +3235,7 @@ static int do_cow_fault(struct vm_fault *vmf)
                return VM_FAULT_OOM;
 
        if (mem_cgroup_try_charge(vmf->cow_page, vma->vm_mm, GFP_KERNEL,
-                               &memcg, false)) {
+                               &vmf->memcg, false)) {
                put_page(vmf->cow_page);
                return VM_FAULT_OOM;
        }
@@ -3245,12 +3243,14 @@ static int do_cow_fault(struct vm_fault *vmf)
        ret = __do_fault(vmf);
        if (unlikely(ret & (VM_FAULT_ERROR | VM_FAULT_NOPAGE | VM_FAULT_RETRY)))
                goto uncharge_out;
+       if (ret & VM_FAULT_DONE_COW)
+               return ret;
 
        if (!(ret & VM_FAULT_DAX_LOCKED))
                copy_user_highpage(vmf->cow_page, vmf->page, vmf->address, vma);
        __SetPageUptodate(vmf->cow_page);
 
-       ret |= alloc_set_pte(vmf, memcg, vmf->cow_page);
+       ret |= alloc_set_pte(vmf, vmf->memcg, vmf->cow_page);
        if (vmf->pte)
                pte_unmap_unlock(vmf->pte, vmf->ptl);
        if (!(ret & VM_FAULT_DAX_LOCKED)) {
@@ -3263,7 +3263,7 @@ static int do_cow_fault(struct vm_fault *vmf)
                goto uncharge_out;
        return ret;
 uncharge_out:
-       mem_cgroup_cancel_charge(vmf->cow_page, memcg, false);
+       mem_cgroup_cancel_charge(vmf->cow_page, vmf->memcg, false);
        put_page(vmf->cow_page);
        return ret;
 }