{
 }
 
+#define __HAVE_ARCH_PTE_SAME
+static inline int pte_same(pte_t pte_a, pte_t pte_b)
+{
+       return pte_val(pte_a) == pte_val(pte_b);
+}
+
+#define __HAVE_ARCH_PTEP_SET_ACCESS_FLAGS
+static inline int ptep_set_access_flags(struct vm_area_struct *vma,
+                                       unsigned long address, pte_t *ptep,
+                                       pte_t entry, int dirty)
+{
+       if (!pte_same(*ptep, entry))
+               set_pte_at(vma->vm_mm, address, ptep, entry);
+       /*
+        * update_mmu_cache will unconditionally execute, handling both
+        * the case that the PTE changed and the spurious fault case.
+        */
+       return true;
+}
+
 /*
  * Conversion functions: convert a page and protection to a page entry,
  * and a page entry and page directory to the page they refer to.
        __update_tlb(vma, address, pte);
 }
 
+#define        __HAVE_ARCH_UPDATE_MMU_TLB
+#define update_mmu_tlb update_mmu_cache
+
 static inline void update_mmu_cache_pmd(struct vm_area_struct *vma,
        unsigned long address, pmd_t *pmdp)
 {
 
 }
 #endif
 
+
+/*
+ * If two threads concurrently fault at the same page, the thread that
+ * won the race updates the PTE and its local TLB/Cache. The other thread
+ * gives up, simply does nothing, and continues; on architectures where
+ * software can update TLB,  local TLB can be updated here to avoid next page
+ * fault. This function updates TLB only, do nothing with cache or others.
+ * It is the difference with function update_mmu_cache.
+ */
+#ifndef __HAVE_ARCH_UPDATE_MMU_TLB
+static inline void update_mmu_tlb(struct vm_area_struct *vma,
+                               unsigned long address, pte_t *ptep)
+{
+}
+#define __HAVE_ARCH_UPDATE_MMU_TLB
+#endif
+
 /*
  * Some architectures may be able to avoid expensive synchronization
  * primitives when modifications are made to PTE's which are already
 
                if (!likely(pte_same(*vmf->pte, vmf->orig_pte))) {
                        /*
                         * Other thread has already handled the fault
-                        * and we don't need to do anything. If it's
-                        * not the case, the fault will be triggered
-                        * again on the same address.
+                        * and update local tlb only
                         */
+                       update_mmu_tlb(vma, addr, vmf->pte);
                        ret = false;
                        goto pte_unlock;
                }
                vmf->pte = pte_offset_map_lock(mm, vmf->pmd, addr, &vmf->ptl);
                locked = true;
                if (!likely(pte_same(*vmf->pte, vmf->orig_pte))) {
-                       /* The PTE changed under us. Retry page fault. */
+                       /* The PTE changed under us, update local tlb */
+                       update_mmu_tlb(vma, addr, vmf->pte);
                        ret = false;
                        goto pte_unlock;
                }
                new_page = old_page;
                page_copied = 1;
        } else {
+               update_mmu_tlb(vma, vmf->address, vmf->pte);
                mem_cgroup_cancel_charge(new_page, memcg, false);
        }
 
         * pte_offset_map_lock.
         */
        if (!pte_same(*vmf->pte, vmf->orig_pte)) {
+               update_mmu_tlb(vmf->vma, vmf->address, vmf->pte);
                pte_unmap_unlock(vmf->pte, vmf->ptl);
                return VM_FAULT_NOPAGE;
        }
                        vmf->pte = pte_offset_map_lock(vma->vm_mm, vmf->pmd,
                                        vmf->address, &vmf->ptl);
                        if (!pte_same(*vmf->pte, vmf->orig_pte)) {
+                               update_mmu_tlb(vma, vmf->address, vmf->pte);
                                unlock_page(vmf->page);
                                pte_unmap_unlock(vmf->pte, vmf->ptl);
                                put_page(vmf->page);
                                                vma->vm_page_prot));
                vmf->pte = pte_offset_map_lock(vma->vm_mm, vmf->pmd,
                                vmf->address, &vmf->ptl);
-               if (!pte_none(*vmf->pte))
+               if (!pte_none(*vmf->pte)) {
+                       update_mmu_tlb(vma, vmf->address, vmf->pte);
                        goto unlock;
+               }
                ret = check_stable_address_space(vma->vm_mm);
                if (ret)
                        goto unlock;
 
        vmf->pte = pte_offset_map_lock(vma->vm_mm, vmf->pmd, vmf->address,
                        &vmf->ptl);
-       if (!pte_none(*vmf->pte))
+       if (!pte_none(*vmf->pte)) {
+               update_mmu_cache(vma, vmf->address, vmf->pte);
                goto release;
+       }
 
        ret = check_stable_address_space(vma->vm_mm);
        if (ret)
        }
 
        /* Re-check under ptl */
-       if (unlikely(!pte_none(*vmf->pte)))
+       if (unlikely(!pte_none(*vmf->pte))) {
+               update_mmu_tlb(vma, vmf->address, vmf->pte);
                return VM_FAULT_NOPAGE;
+       }
 
        flush_icache_page(vma, page);
        entry = mk_pte(page, vma->vm_page_prot);
        vmf->ptl = pte_lockptr(vmf->vma->vm_mm, vmf->pmd);
        spin_lock(vmf->ptl);
        entry = vmf->orig_pte;
-       if (unlikely(!pte_same(*vmf->pte, entry)))
+       if (unlikely(!pte_same(*vmf->pte, entry))) {
+               update_mmu_tlb(vmf->vma, vmf->address, vmf->pte);
                goto unlock;
+       }
        if (vmf->flags & FAULT_FLAG_WRITE) {
                if (!pte_write(entry))
                        return do_wp_page(vmf);