aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/mm/pageattr.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86/mm/pageattr.c')
-rw-r--r--arch/x86/mm/pageattr.c34
1 files changed, 15 insertions, 19 deletions
diff --git a/arch/x86/mm/pageattr.c b/arch/x86/mm/pageattr.c
index 029fb07b3f03..fb2eedba76ad 100644
--- a/arch/x86/mm/pageattr.c
+++ b/arch/x86/mm/pageattr.c
@@ -27,11 +27,6 @@ struct cpa_data {
27 int flushtlb; 27 int flushtlb;
28}; 28};
29 29
30enum {
31 CPA_NO_SPLIT = 0,
32 CPA_SPLIT,
33};
34
35static inline int 30static inline int
36within(unsigned long addr, unsigned long start, unsigned long end) 31within(unsigned long addr, unsigned long start, unsigned long end)
37{ 32{
@@ -263,7 +258,7 @@ try_preserve_large_page(pte_t *kpte, unsigned long address,
263 unsigned long nextpage_addr, numpages, pmask, psize, flags; 258 unsigned long nextpage_addr, numpages, pmask, psize, flags;
264 pte_t new_pte, old_pte, *tmp; 259 pte_t new_pte, old_pte, *tmp;
265 pgprot_t old_prot, new_prot; 260 pgprot_t old_prot, new_prot;
266 int level, res = CPA_SPLIT; 261 int level, do_split = 1;
267 262
268 /* 263 /*
269 * An Athlon 64 X2 showed hard hangs if we tried to preserve 264 * An Athlon 64 X2 showed hard hangs if we tried to preserve
@@ -274,7 +269,7 @@ try_preserve_large_page(pte_t *kpte, unsigned long address,
274 * disable this code until the hang can be debugged: 269 * disable this code until the hang can be debugged:
275 */ 270 */
276 if (boot_cpu_data.x86_vendor == X86_VENDOR_AMD) 271 if (boot_cpu_data.x86_vendor == X86_VENDOR_AMD)
277 return res; 272 return 1;
278 273
279 spin_lock_irqsave(&pgd_lock, flags); 274 spin_lock_irqsave(&pgd_lock, flags);
280 /* 275 /*
@@ -297,7 +292,7 @@ try_preserve_large_page(pte_t *kpte, unsigned long address,
297 break; 292 break;
298#endif 293#endif
299 default: 294 default:
300 res = -EINVAL; 295 do_split = -EINVAL;
301 goto out_unlock; 296 goto out_unlock;
302 } 297 }
303 298
@@ -325,7 +320,7 @@ try_preserve_large_page(pte_t *kpte, unsigned long address,
325 * above: 320 * above:
326 */ 321 */
327 if (pgprot_val(new_prot) == pgprot_val(old_prot)) { 322 if (pgprot_val(new_prot) == pgprot_val(old_prot)) {
328 res = CPA_NO_SPLIT; 323 do_split = 0;
329 goto out_unlock; 324 goto out_unlock;
330 } 325 }
331 326
@@ -345,13 +340,13 @@ try_preserve_large_page(pte_t *kpte, unsigned long address,
345 new_pte = pfn_pte(pte_pfn(old_pte), canon_pgprot(new_prot)); 340 new_pte = pfn_pte(pte_pfn(old_pte), canon_pgprot(new_prot));
346 __set_pmd_pte(kpte, address, new_pte); 341 __set_pmd_pte(kpte, address, new_pte);
347 cpa->flushtlb = 1; 342 cpa->flushtlb = 1;
348 res = CPA_NO_SPLIT; 343 do_split = 0;
349 } 344 }
350 345
351out_unlock: 346out_unlock:
352 spin_unlock_irqrestore(&pgd_lock, flags); 347 spin_unlock_irqrestore(&pgd_lock, flags);
353 348
354 return res; 349 return do_split;
355} 350}
356 351
357static int split_large_page(pte_t *kpte, unsigned long address) 352static int split_large_page(pte_t *kpte, unsigned long address)
@@ -429,7 +424,7 @@ out_unlock:
429static int __change_page_attr(unsigned long address, struct cpa_data *cpa) 424static int __change_page_attr(unsigned long address, struct cpa_data *cpa)
430{ 425{
431 struct page *kpte_page; 426 struct page *kpte_page;
432 int level, res; 427 int level, do_split;
433 pte_t *kpte; 428 pte_t *kpte;
434 429
435repeat: 430repeat:
@@ -480,25 +475,26 @@ repeat:
480 * Check, whether we can keep the large page intact 475 * Check, whether we can keep the large page intact
481 * and just change the pte: 476 * and just change the pte:
482 */ 477 */
483 res = try_preserve_large_page(kpte, address, cpa); 478 do_split = try_preserve_large_page(kpte, address, cpa);
484 if (res < 0) 479 if (do_split < 0)
485 return res; 480 return do_split;
486 481
487 /* 482 /*
488 * When the range fits into the existing large page, 483 * When the range fits into the existing large page,
489 * return. cp->numpages and cpa->tlbflush have been updated in 484 * return. cp->numpages and cpa->tlbflush have been updated in
490 * try_large_page: 485 * try_large_page:
491 */ 486 */
492 if (res == CPA_NO_SPLIT) 487 if (do_split == 0)
493 return 0; 488 return 0;
494 489
495 /* 490 /*
496 * We have to split the large page: 491 * We have to split the large page:
497 */ 492 */
498 res = split_large_page(kpte, address); 493 do_split = split_large_page(kpte, address);
499 if (res) 494 if (do_split)
500 return res; 495 return do_split;
501 cpa->flushtlb = 1; 496 cpa->flushtlb = 1;
497
502 goto repeat; 498 goto repeat;
503} 499}
504 500