diff options
Diffstat (limited to 'arch/arm64/include/asm/pgtable.h')
| -rw-r--r-- | arch/arm64/include/asm/pgtable.h | 23 | 
1 files changed, 13 insertions, 10 deletions
| diff --git a/arch/arm64/include/asm/pgtable.h b/arch/arm64/include/asm/pgtable.h index 094374c82db0..7e2c27e63cd8 100644 --- a/arch/arm64/include/asm/pgtable.h +++ b/arch/arm64/include/asm/pgtable.h @@ -218,7 +218,7 @@ static inline pmd_t pmd_mkcont(pmd_t pmd)  static inline void set_pte(pte_t *ptep, pte_t pte)  { -	*ptep = pte; +	WRITE_ONCE(*ptep, pte);  	/*  	 * Only if the new pte is valid and kernel, otherwise TLB maintenance @@ -250,6 +250,8 @@ extern void __sync_icache_dcache(pte_t pteval, unsigned long addr);  static inline void set_pte_at(struct mm_struct *mm, unsigned long addr,  			      pte_t *ptep, pte_t pte)  { +	pte_t old_pte; +  	if (pte_present(pte) && pte_user_exec(pte) && !pte_special(pte))  		__sync_icache_dcache(pte, addr); @@ -258,14 +260,15 @@ static inline void set_pte_at(struct mm_struct *mm, unsigned long addr,  	 * hardware updates of the pte (ptep_set_access_flags safely changes  	 * valid ptes without going through an invalid entry).  	 */ -	if (IS_ENABLED(CONFIG_DEBUG_VM) && pte_valid(*ptep) && pte_valid(pte) && +	old_pte = READ_ONCE(*ptep); +	if (IS_ENABLED(CONFIG_DEBUG_VM) && pte_valid(old_pte) && pte_valid(pte) &&  	   (mm == current->active_mm || atomic_read(&mm->mm_users) > 1)) {  		VM_WARN_ONCE(!pte_young(pte),  			     "%s: racy access flag clearing: 0x%016llx -> 0x%016llx", -			     __func__, pte_val(*ptep), pte_val(pte)); -		VM_WARN_ONCE(pte_write(*ptep) && !pte_dirty(pte), +			     __func__, pte_val(old_pte), pte_val(pte)); +		VM_WARN_ONCE(pte_write(old_pte) && !pte_dirty(pte),  			     "%s: racy dirty state clearing: 0x%016llx -> 0x%016llx", -			     __func__, pte_val(*ptep), pte_val(pte)); +			     __func__, pte_val(old_pte), pte_val(pte));  	}  	set_pte(ptep, pte); @@ -431,7 +434,7 @@ extern pgprot_t phys_mem_access_prot(struct file *file, unsigned long pfn,  static inline void set_pmd(pmd_t *pmdp, pmd_t pmd)  { -	*pmdp = pmd; +	WRITE_ONCE(*pmdp, pmd);  	dsb(ishst);  	isb();  } @@ -482,7 +485,7 @@ static inline phys_addr_t pmd_page_paddr(pmd_t pmd)  static inline void set_pud(pud_t *pudp, pud_t pud)  { -	*pudp = pud; +	WRITE_ONCE(*pudp, pud);  	dsb(ishst);  	isb();  } @@ -500,7 +503,7 @@ static inline phys_addr_t pud_page_paddr(pud_t pud)  /* Find an entry in the second-level page table. */  #define pmd_index(addr)		(((addr) >> PMD_SHIFT) & (PTRS_PER_PMD - 1)) -#define pmd_offset_phys(dir, addr)	(pud_page_paddr(*(dir)) + pmd_index(addr) * sizeof(pmd_t)) +#define pmd_offset_phys(dir, addr)	(pud_page_paddr(READ_ONCE(*(dir))) + pmd_index(addr) * sizeof(pmd_t))  #define pmd_offset(dir, addr)		((pmd_t *)__va(pmd_offset_phys((dir), (addr))))  #define pmd_set_fixmap(addr)		((pmd_t *)set_fixmap_offset(FIX_PMD, addr)) @@ -535,7 +538,7 @@ static inline phys_addr_t pud_page_paddr(pud_t pud)  static inline void set_pgd(pgd_t *pgdp, pgd_t pgd)  { -	*pgdp = pgd; +	WRITE_ONCE(*pgdp, pgd);  	dsb(ishst);  } @@ -552,7 +555,7 @@ static inline phys_addr_t pgd_page_paddr(pgd_t pgd)  /* Find an entry in the frst-level page table. */  #define pud_index(addr)		(((addr) >> PUD_SHIFT) & (PTRS_PER_PUD - 1)) -#define pud_offset_phys(dir, addr)	(pgd_page_paddr(*(dir)) + pud_index(addr) * sizeof(pud_t)) +#define pud_offset_phys(dir, addr)	(pgd_page_paddr(READ_ONCE(*(dir))) + pud_index(addr) * sizeof(pud_t))  #define pud_offset(dir, addr)		((pud_t *)__va(pud_offset_phys((dir), (addr))))  #define pud_set_fixmap(addr)		((pud_t *)set_fixmap_offset(FIX_PUD, addr)) | 
