diff options
Diffstat (limited to 'arch/x86/mm/pageattr.c')
-rw-r--r-- | arch/x86/mm/pageattr.c | 13 |
1 files changed, 8 insertions, 5 deletions
diff --git a/arch/x86/mm/pageattr.c b/arch/x86/mm/pageattr.c index effcd78d5f40..d18c41d752f3 100644 --- a/arch/x86/mm/pageattr.c +++ b/arch/x86/mm/pageattr.c | |||
@@ -3,6 +3,7 @@ | |||
3 | * Thanks to Ben LaHaise for precious feedback. | 3 | * Thanks to Ben LaHaise for precious feedback. |
4 | */ | 4 | */ |
5 | #include <linux/highmem.h> | 5 | #include <linux/highmem.h> |
6 | #include <linux/bootmem.h> | ||
6 | #include <linux/module.h> | 7 | #include <linux/module.h> |
7 | #include <linux/sched.h> | 8 | #include <linux/sched.h> |
8 | #include <linux/slab.h> | 9 | #include <linux/slab.h> |
@@ -144,13 +145,15 @@ out_unlock: | |||
144 | } | 145 | } |
145 | 146 | ||
146 | static int | 147 | static int |
147 | __change_page_attr(unsigned long address, struct page *page, pgprot_t prot) | 148 | __change_page_attr(unsigned long address, unsigned long pfn, pgprot_t prot) |
148 | { | 149 | { |
149 | struct page *kpte_page; | 150 | struct page *kpte_page; |
150 | int level, err = 0; | 151 | int level, err = 0; |
151 | pte_t *kpte; | 152 | pte_t *kpte; |
152 | 153 | ||
153 | BUG_ON(PageHighMem(page)); | 154 | #ifdef CONFIG_X86_32 |
155 | BUG_ON(pfn > max_low_pfn); | ||
156 | #endif | ||
154 | 157 | ||
155 | repeat: | 158 | repeat: |
156 | kpte = lookup_address(address, &level); | 159 | kpte = lookup_address(address, &level); |
@@ -164,7 +167,7 @@ repeat: | |||
164 | prot = check_exec(prot, address); | 167 | prot = check_exec(prot, address); |
165 | 168 | ||
166 | if (level == PG_LEVEL_4K) { | 169 | if (level == PG_LEVEL_4K) { |
167 | set_pte_atomic(kpte, mk_pte(page, canon_pgprot(prot))); | 170 | set_pte_atomic(kpte, pfn_pte(pfn, canon_pgprot(prot))); |
168 | } else { | 171 | } else { |
169 | err = split_large_page(kpte, address); | 172 | err = split_large_page(kpte, address); |
170 | if (!err) | 173 | if (!err) |
@@ -203,7 +206,7 @@ int change_page_attr_addr(unsigned long address, int numpages, pgprot_t prot) | |||
203 | unsigned long pfn = __pa(address) >> PAGE_SHIFT; | 206 | unsigned long pfn = __pa(address) >> PAGE_SHIFT; |
204 | 207 | ||
205 | if (!kernel_map || pte_present(pfn_pte(0, prot))) { | 208 | if (!kernel_map || pte_present(pfn_pte(0, prot))) { |
206 | err = __change_page_attr(address, pfn_to_page(pfn), prot); | 209 | err = __change_page_attr(address, pfn, prot); |
207 | if (err) | 210 | if (err) |
208 | break; | 211 | break; |
209 | } | 212 | } |
@@ -219,7 +222,7 @@ int change_page_attr_addr(unsigned long address, int numpages, pgprot_t prot) | |||
219 | addr2 = __START_KERNEL_map + __pa(address); | 222 | addr2 = __START_KERNEL_map + __pa(address); |
220 | /* Make sure the kernel mappings stay executable */ | 223 | /* Make sure the kernel mappings stay executable */ |
221 | prot2 = pte_pgprot(pte_mkexec(pfn_pte(0, prot))); | 224 | prot2 = pte_pgprot(pte_mkexec(pfn_pte(0, prot))); |
222 | err = __change_page_attr(addr2, pfn_to_page(pfn), prot2); | 225 | err = __change_page_attr(addr2, pfn, prot2); |
223 | } | 226 | } |
224 | #endif | 227 | #endif |
225 | } | 228 | } |