diff options
Diffstat (limited to 'arch/powerpc/mm/gup.c')
| -rw-r--r-- | arch/powerpc/mm/gup.c | 18 | 
1 files changed, 12 insertions, 6 deletions
| diff --git a/arch/powerpc/mm/gup.c b/arch/powerpc/mm/gup.c index 4b921affa495..49822d90ea96 100644 --- a/arch/powerpc/mm/gup.c +++ b/arch/powerpc/mm/gup.c @@ -34,7 +34,7 @@ static noinline int gup_pte_range(pmd_t pmd, unsigned long addr,  	ptep = pte_offset_kernel(&pmd, addr);  	do { -		pte_t pte = *ptep; +		pte_t pte = ACCESS_ONCE(*ptep);  		struct page *page;  		if ((pte_val(pte) & mask) != result) @@ -63,12 +63,18 @@ static int gup_pmd_range(pud_t pud, unsigned long addr, unsigned long end,  	pmdp = pmd_offset(&pud, addr);  	do { -		pmd_t pmd = *pmdp; +		pmd_t pmd = ACCESS_ONCE(*pmdp);  		next = pmd_addr_end(addr, end); -		if (pmd_none(pmd)) +		/* +		 * If we find a splitting transparent hugepage we +		 * return zero. That will result in taking the slow +		 * path which will call wait_split_huge_page() +		 * if the pmd is still in splitting state +		 */ +		if (pmd_none(pmd) || pmd_trans_splitting(pmd))  			return 0; -		if (pmd_huge(pmd)) { +		if (pmd_huge(pmd) || pmd_large(pmd)) {  			if (!gup_hugepte((pte_t *)pmdp, PMD_SIZE, addr, next,  					 write, pages, nr))  				return 0; @@ -91,7 +97,7 @@ static int gup_pud_range(pgd_t pgd, unsigned long addr, unsigned long end,  	pudp = pud_offset(&pgd, addr);  	do { -		pud_t pud = *pudp; +		pud_t pud = ACCESS_ONCE(*pudp);  		next = pud_addr_end(addr, end);  		if (pud_none(pud)) @@ -154,7 +160,7 @@ int get_user_pages_fast(unsigned long start, int nr_pages, int write,  	pgdp = pgd_offset(mm, addr);  	do { -		pgd_t pgd = *pgdp; +		pgd_t pgd = ACCESS_ONCE(*pgdp);  		pr_devel("  %016lx: normal pgd %p\n", addr,  			 (void *)pgd_val(pgd)); |