aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86')
-rw-r--r--arch/x86/mm/pageattr.c50
1 files changed, 23 insertions, 27 deletions
diff --git a/arch/x86/mm/pageattr.c b/arch/x86/mm/pageattr.c
index 3cfe9ced8a4c..2ab058b0947d 100644
--- a/arch/x86/mm/pageattr.c
+++ b/arch/x86/mm/pageattr.c
@@ -681,8 +681,9 @@ static int __change_page_attr_set_clr(struct cpa_data *cpa, int checkalias);
681static int cpa_process_alias(struct cpa_data *cpa) 681static int cpa_process_alias(struct cpa_data *cpa)
682{ 682{
683 struct cpa_data alias_cpa; 683 struct cpa_data alias_cpa;
684 int ret = 0; 684 unsigned long laddr = (unsigned long)__va(cpa->pfn << PAGE_SHIFT);
685 unsigned long temp_cpa_vaddr, vaddr; 685 unsigned long vaddr;
686 int ret;
686 687
687 if (cpa->pfn >= max_pfn_mapped) 688 if (cpa->pfn >= max_pfn_mapped)
688 return 0; 689 return 0;
@@ -706,42 +707,37 @@ static int cpa_process_alias(struct cpa_data *cpa)
706 PAGE_OFFSET + (max_pfn_mapped << PAGE_SHIFT)))) { 707 PAGE_OFFSET + (max_pfn_mapped << PAGE_SHIFT)))) {
707 708
708 alias_cpa = *cpa; 709 alias_cpa = *cpa;
709 temp_cpa_vaddr = (unsigned long) __va(cpa->pfn << PAGE_SHIFT); 710 alias_cpa.vaddr = &laddr;
710 alias_cpa.vaddr = &temp_cpa_vaddr;
711 alias_cpa.flags &= ~(CPA_PAGES_ARRAY | CPA_ARRAY); 711 alias_cpa.flags &= ~(CPA_PAGES_ARRAY | CPA_ARRAY);
712 712
713
714 ret = __change_page_attr_set_clr(&alias_cpa, 0); 713 ret = __change_page_attr_set_clr(&alias_cpa, 0);
714 if (ret)
715 return ret;
715 } 716 }
716 717
717#ifdef CONFIG_X86_64 718#ifdef CONFIG_X86_64
718 if (ret)
719 return ret;
720 /* 719 /*
721 * No need to redo, when the primary call touched the high 720 * If the primary call didn't touch the high mapping already
722 * mapping already: 721 * and the physical address is inside the kernel map, we need
723 */
724 if (within(vaddr, (unsigned long) _text, _brk_end))
725 return 0;
726
727 /*
728 * If the physical address is inside the kernel map, we need
729 * to touch the high mapped kernel as well: 722 * to touch the high mapped kernel as well:
730 */ 723 */
731 if (!within(cpa->pfn, highmap_start_pfn(), highmap_end_pfn())) 724 if (!within(vaddr, (unsigned long)_text, _brk_end) &&
732 return 0; 725 within(cpa->pfn, highmap_start_pfn(), highmap_end_pfn())) {
733 726 unsigned long temp_cpa_vaddr = (cpa->pfn << PAGE_SHIFT) +
734 alias_cpa = *cpa; 727 __START_KERNEL_map - phys_base;
735 temp_cpa_vaddr = (cpa->pfn << PAGE_SHIFT) + __START_KERNEL_map - phys_base; 728 alias_cpa = *cpa;
736 alias_cpa.vaddr = &temp_cpa_vaddr; 729 alias_cpa.vaddr = &temp_cpa_vaddr;
737 alias_cpa.flags &= ~(CPA_PAGES_ARRAY | CPA_ARRAY); 730 alias_cpa.flags &= ~(CPA_PAGES_ARRAY | CPA_ARRAY);
738 731
739 /* 732 /*
740 * The high mapping range is imprecise, so ignore the return value. 733 * The high mapping range is imprecise, so ignore the
741 */ 734 * return value.
742 __change_page_attr_set_clr(&alias_cpa, 0); 735 */
736 __change_page_attr_set_clr(&alias_cpa, 0);
737 }
743#endif 738#endif
744 return ret; 739
740 return 0;
745} 741}
746 742
747static int __change_page_attr_set_clr(struct cpa_data *cpa, int checkalias) 743static int __change_page_attr_set_clr(struct cpa_data *cpa, int checkalias)