We need an atomic way to make pmd page table entry not-present. This is required to implement pmdp_invalidate() that doesn't loose dirty or access bits. On x86, we need to clear two bits -- _PAGE_PRESENT and _PAGE_PROTNONE -- to make the entry non-present. The implementation uses cmpxchg() loop to make it atomically. PAE requires special treatment to avoid expensive cmpxchg8b(). Both bits are in the lower part of the entry, so we can use 4-byte cmpxchg() on this part of page table entry. Signed-off-by: Kirill A. Shutemov <kirill.shutemov@xxxxxxxxxxxxxxx> Cc: Ingo Molnar <mingo@xxxxxxxxxx> Cc: H. Peter Anvin <hpa@xxxxxxxxx> Cc: Thomas Gleixner <tglx@xxxxxxxxxxxxx> --- arch/x86/include/asm/pgtable-3level.h | 17 +++++++++++++++++ arch/x86/include/asm/pgtable.h | 13 +++++++++++++ 2 files changed, 30 insertions(+) diff --git a/arch/x86/include/asm/pgtable-3level.h b/arch/x86/include/asm/pgtable-3level.h index 50d35e3185f5..b6efa955ecd0 100644 --- a/arch/x86/include/asm/pgtable-3level.h +++ b/arch/x86/include/asm/pgtable-3level.h @@ -176,8 +176,25 @@ static inline pmd_t native_pmdp_get_and_clear(pmd_t *pmdp) return res.pmd; } + +#define pmdp_mknotpresent pmdp_mknotpresent +static inline void pmdp_mknotpresent(pmd_t *pmdp) +{ + union split_pmd *p, old, new; + + p = (union split_pmd *)pmdp; + { + old = *p; + new.pmd = pmd_mknotpresent(old.pmd); + } while (cmpxchg(&p->pmd_low, old.pmd_low, new.pmd_low) != old.pmd_low); +} #else #define native_pmdp_get_and_clear(xp) native_local_pmdp_get_and_clear(xp) + +static inline void pmdp_mknotpresent(pmd_t *pmdp) +{ + *pmdp = pmd_mknotpresent(*pmdp); +} #endif #ifdef CONFIG_SMP diff --git a/arch/x86/include/asm/pgtable.h b/arch/x86/include/asm/pgtable.h index f5af95a0c6b8..576420df12b8 100644 --- a/arch/x86/include/asm/pgtable.h +++ b/arch/x86/include/asm/pgtable.h @@ -1092,6 +1092,19 @@ static inline void pmdp_set_wrprotect(struct mm_struct *mm, clear_bit(_PAGE_BIT_RW, (unsigned long *)pmdp); } +#ifndef pmdp_mknotpresent +#define pmdp_mknotpresent pmdp_mknotpresent +static inline void pmdp_mknotpresent(pmd_t *pmdp) +{ + pmd_t old, new; + + { + old = *pmdp; + new = pmd_mknotpresent(old); + } while (pmd_val(cmpxchg(pmdp, old, new)) != pmd_val(old)); +} +#endif + /* * clone_pgd_range(pgd_t *dst, pgd_t *src, int count); * -- 2.11.0