mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-18 22:14:16 +00:00 
			
		
		
		
	sparc64: Fix regression in pmdp_invalidate().
pmdp_invalidate() was changed to update the pmd atomically
(to not lose dirty/access bits) and return the original pmd
value.
However, in doing so, we lost a lot of the essential work that
set_pmd_at() does, namely to update hugepage mapping counts and
queuing up the batched TLB flush entry.
Thus we were not flushing entries out of the TLB when making
such PMD changes.
Fix this by abstracting the accounting work of set_pmd_at() out into a
separate function, and call it from pmdp_establish().
Fixes: a8e654f01c ("sparc64: update pmdp_invalidate() to return old pmd value")
Signed-off-by: David S. Miller <davem@davemloft.net>
			
			
This commit is contained in:
		
							parent
							
								
									e2c15aff5f
								
							
						
					
					
						commit
						cfb61b5e3e
					
				
					 1 changed files with 13 additions and 6 deletions
				
			
		|  | @ -163,13 +163,10 @@ static void tlb_batch_pmd_scan(struct mm_struct *mm, unsigned long vaddr, | |||
| 	pte_unmap(pte); | ||||
| } | ||||
| 
 | ||||
| void set_pmd_at(struct mm_struct *mm, unsigned long addr, | ||||
| 		pmd_t *pmdp, pmd_t pmd) | ||||
| 
 | ||||
| static void __set_pmd_acct(struct mm_struct *mm, unsigned long addr, | ||||
| 			   pmd_t orig, pmd_t pmd) | ||||
| { | ||||
| 	pmd_t orig = *pmdp; | ||||
| 
 | ||||
| 	*pmdp = pmd; | ||||
| 
 | ||||
| 	if (mm == &init_mm) | ||||
| 		return; | ||||
| 
 | ||||
|  | @ -219,6 +216,15 @@ void set_pmd_at(struct mm_struct *mm, unsigned long addr, | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| void set_pmd_at(struct mm_struct *mm, unsigned long addr, | ||||
| 		pmd_t *pmdp, pmd_t pmd) | ||||
| { | ||||
| 	pmd_t orig = *pmdp; | ||||
| 
 | ||||
| 	*pmdp = pmd; | ||||
| 	__set_pmd_acct(mm, addr, orig, pmd); | ||||
| } | ||||
| 
 | ||||
| static inline pmd_t pmdp_establish(struct vm_area_struct *vma, | ||||
| 		unsigned long address, pmd_t *pmdp, pmd_t pmd) | ||||
| { | ||||
|  | @ -227,6 +233,7 @@ static inline pmd_t pmdp_establish(struct vm_area_struct *vma, | |||
| 	do { | ||||
| 		old = *pmdp; | ||||
| 	} while (cmpxchg64(&pmdp->pmd, old.pmd, pmd.pmd) != old.pmd); | ||||
| 	__set_pmd_acct(vma->vm_mm, address, old, pmd); | ||||
| 
 | ||||
| 	return old; | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		
		Reference in a new issue
	
	 David S. Miller
						David S. Miller