diff options
Diffstat (limited to 'arch/x86')
-rw-r--r-- | arch/x86/mm/pageattr.c | 20 |
1 files changed, 16 insertions, 4 deletions
diff --git a/arch/x86/mm/pageattr.c b/arch/x86/mm/pageattr.c index 270cab2e6030..7d9517abc9af 100644 --- a/arch/x86/mm/pageattr.c +++ b/arch/x86/mm/pageattr.c | |||
@@ -31,6 +31,7 @@ struct cpa_data { | |||
31 | int numpages; | 31 | int numpages; |
32 | int flushtlb; | 32 | int flushtlb; |
33 | unsigned long pfn; | 33 | unsigned long pfn; |
34 | unsigned force_split : 1; | ||
34 | }; | 35 | }; |
35 | 36 | ||
36 | #ifdef CONFIG_X86_64 | 37 | #ifdef CONFIG_X86_64 |
@@ -262,6 +263,9 @@ try_preserve_large_page(pte_t *kpte, unsigned long address, | |||
262 | int i, do_split = 1; | 263 | int i, do_split = 1; |
263 | unsigned int level; | 264 | unsigned int level; |
264 | 265 | ||
266 | if (cpa->force_split) | ||
267 | return 1; | ||
268 | |||
265 | spin_lock_irqsave(&pgd_lock, flags); | 269 | spin_lock_irqsave(&pgd_lock, flags); |
266 | /* | 270 | /* |
267 | * Check for races, another CPU might have split this page | 271 | * Check for races, another CPU might have split this page |
@@ -696,7 +700,8 @@ static inline int cache_attr(pgprot_t attr) | |||
696 | } | 700 | } |
697 | 701 | ||
698 | static int change_page_attr_set_clr(unsigned long addr, int numpages, | 702 | static int change_page_attr_set_clr(unsigned long addr, int numpages, |
699 | pgprot_t mask_set, pgprot_t mask_clr) | 703 | pgprot_t mask_set, pgprot_t mask_clr, |
704 | int force_split) | ||
700 | { | 705 | { |
701 | struct cpa_data cpa; | 706 | struct cpa_data cpa; |
702 | int ret, cache, checkalias; | 707 | int ret, cache, checkalias; |
@@ -707,7 +712,7 @@ static int change_page_attr_set_clr(unsigned long addr, int numpages, | |||
707 | */ | 712 | */ |
708 | mask_set = canon_pgprot(mask_set); | 713 | mask_set = canon_pgprot(mask_set); |
709 | mask_clr = canon_pgprot(mask_clr); | 714 | mask_clr = canon_pgprot(mask_clr); |
710 | if (!pgprot_val(mask_set) && !pgprot_val(mask_clr)) | 715 | if (!pgprot_val(mask_set) && !pgprot_val(mask_clr) && !force_split) |
711 | return 0; | 716 | return 0; |
712 | 717 | ||
713 | /* Ensure we are PAGE_SIZE aligned */ | 718 | /* Ensure we are PAGE_SIZE aligned */ |
@@ -724,6 +729,7 @@ static int change_page_attr_set_clr(unsigned long addr, int numpages, | |||
724 | cpa.mask_set = mask_set; | 729 | cpa.mask_set = mask_set; |
725 | cpa.mask_clr = mask_clr; | 730 | cpa.mask_clr = mask_clr; |
726 | cpa.flushtlb = 0; | 731 | cpa.flushtlb = 0; |
732 | cpa.force_split = force_split; | ||
727 | 733 | ||
728 | /* No alias checking for _NX bit modifications */ | 734 | /* No alias checking for _NX bit modifications */ |
729 | checkalias = (pgprot_val(mask_set) | pgprot_val(mask_clr)) != _PAGE_NX; | 735 | checkalias = (pgprot_val(mask_set) | pgprot_val(mask_clr)) != _PAGE_NX; |
@@ -762,13 +768,13 @@ out: | |||
762 | static inline int change_page_attr_set(unsigned long addr, int numpages, | 768 | static inline int change_page_attr_set(unsigned long addr, int numpages, |
763 | pgprot_t mask) | 769 | pgprot_t mask) |
764 | { | 770 | { |
765 | return change_page_attr_set_clr(addr, numpages, mask, __pgprot(0)); | 771 | return change_page_attr_set_clr(addr, numpages, mask, __pgprot(0), 0); |
766 | } | 772 | } |
767 | 773 | ||
768 | static inline int change_page_attr_clear(unsigned long addr, int numpages, | 774 | static inline int change_page_attr_clear(unsigned long addr, int numpages, |
769 | pgprot_t mask) | 775 | pgprot_t mask) |
770 | { | 776 | { |
771 | return change_page_attr_set_clr(addr, numpages, __pgprot(0), mask); | 777 | return change_page_attr_set_clr(addr, numpages, __pgprot(0), mask, 0); |
772 | } | 778 | } |
773 | 779 | ||
774 | int _set_memory_uc(unsigned long addr, int numpages) | 780 | int _set_memory_uc(unsigned long addr, int numpages) |
@@ -847,6 +853,12 @@ int set_memory_np(unsigned long addr, int numpages) | |||
847 | return change_page_attr_clear(addr, numpages, __pgprot(_PAGE_PRESENT)); | 853 | return change_page_attr_clear(addr, numpages, __pgprot(_PAGE_PRESENT)); |
848 | } | 854 | } |
849 | 855 | ||
856 | int set_memory_4k(unsigned long addr, int numpages) | ||
857 | { | ||
858 | return change_page_attr_set_clr(addr, numpages, __pgprot(0), | ||
859 | __pgprot(0), 1); | ||
860 | } | ||
861 | |||
850 | int set_pages_uc(struct page *page, int numpages) | 862 | int set_pages_uc(struct page *page, int numpages) |
851 | { | 863 | { |
852 | unsigned long addr = (unsigned long)page_address(page); | 864 | unsigned long addr = (unsigned long)page_address(page); |