diff options
author | Ingo Molnar <mingo@elte.hu> | 2008-01-30 07:33:43 -0500 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-01-30 07:33:43 -0500 |
commit | f0646e43acb18f0e00b00085dc88bc3f403e7930 (patch) | |
tree | c47968a44ac541854929f03d22cf2fb76d76cb55 | |
parent | a5a5dc31794c3271c066835ad2c90c58a3805569 (diff) |
x86: return the page table level in lookup_address()
based on this patch from Andi Kleen:
| Subject: CPA: Return the page table level in lookup_address()
| From: Andi Kleen <ak@suse.de>
|
| Needed for the next change.
|
| And change all the callers.
and ported it to x86.git.
Signed-off-by: Andi Kleen <ak@suse.de>
Acked-by: Jan Beulich <jbeulich@novell.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
-rw-r--r-- | arch/x86/mm/fault_32.c | 3 | ||||
-rw-r--r-- | arch/x86/mm/init_32.c | 3 | ||||
-rw-r--r-- | arch/x86/mm/pageattr_32.c | 7 | ||||
-rw-r--r-- | arch/x86/mm/pageattr_64.c | 7 | ||||
-rw-r--r-- | arch/x86/xen/mmu.c | 9 | ||||
-rw-r--r-- | include/asm-x86/pgtable_32.h | 2 | ||||
-rw-r--r-- | include/asm-x86/pgtable_64.h | 2 |
7 files changed, 22 insertions, 11 deletions
diff --git a/arch/x86/mm/fault_32.c b/arch/x86/mm/fault_32.c index f7972ae7da0..f4f8c324715 100644 --- a/arch/x86/mm/fault_32.c +++ b/arch/x86/mm/fault_32.c | |||
@@ -613,7 +613,8 @@ no_context: | |||
613 | 613 | ||
614 | #ifdef CONFIG_X86_PAE | 614 | #ifdef CONFIG_X86_PAE |
615 | if (error_code & PF_INSTR) { | 615 | if (error_code & PF_INSTR) { |
616 | pte_t *pte = lookup_address(address); | 616 | int level; |
617 | pte_t *pte = lookup_address(address, &level); | ||
617 | 618 | ||
618 | if (pte && pte_present(*pte) && !pte_exec(*pte)) | 619 | if (pte && pte_present(*pte) && !pte_exec(*pte)) |
619 | printk(KERN_CRIT "kernel tried to execute " | 620 | printk(KERN_CRIT "kernel tried to execute " |
diff --git a/arch/x86/mm/init_32.c b/arch/x86/mm/init_32.c index 5080646da77..206e3f6800b 100644 --- a/arch/x86/mm/init_32.c +++ b/arch/x86/mm/init_32.c | |||
@@ -535,11 +535,12 @@ int __init set_kernel_exec(unsigned long vaddr, int enable) | |||
535 | { | 535 | { |
536 | pte_t *pte; | 536 | pte_t *pte; |
537 | int ret = 1; | 537 | int ret = 1; |
538 | int level; | ||
538 | 539 | ||
539 | if (!nx_enabled) | 540 | if (!nx_enabled) |
540 | goto out; | 541 | goto out; |
541 | 542 | ||
542 | pte = lookup_address(vaddr); | 543 | pte = lookup_address(vaddr, &level); |
543 | BUG_ON(!pte); | 544 | BUG_ON(!pte); |
544 | 545 | ||
545 | if (!pte_exec(*pte)) | 546 | if (!pte_exec(*pte)) |
diff --git a/arch/x86/mm/pageattr_32.c b/arch/x86/mm/pageattr_32.c index be4656403d7..523fd5b37df 100644 --- a/arch/x86/mm/pageattr_32.c +++ b/arch/x86/mm/pageattr_32.c | |||
@@ -18,7 +18,7 @@ | |||
18 | static DEFINE_SPINLOCK(cpa_lock); | 18 | static DEFINE_SPINLOCK(cpa_lock); |
19 | static struct list_head df_list = LIST_HEAD_INIT(df_list); | 19 | static struct list_head df_list = LIST_HEAD_INIT(df_list); |
20 | 20 | ||
21 | pte_t *lookup_address(unsigned long address) | 21 | pte_t *lookup_address(unsigned long address, int *level) |
22 | { | 22 | { |
23 | pgd_t *pgd = pgd_offset_k(address); | 23 | pgd_t *pgd = pgd_offset_k(address); |
24 | pud_t *pud; | 24 | pud_t *pud; |
@@ -32,8 +32,10 @@ pte_t *lookup_address(unsigned long address) | |||
32 | pmd = pmd_offset(pud, address); | 32 | pmd = pmd_offset(pud, address); |
33 | if (pmd_none(*pmd)) | 33 | if (pmd_none(*pmd)) |
34 | return NULL; | 34 | return NULL; |
35 | *level = 2; | ||
35 | if (pmd_large(*pmd)) | 36 | if (pmd_large(*pmd)) |
36 | return (pte_t *)pmd; | 37 | return (pte_t *)pmd; |
38 | *level = 3; | ||
37 | 39 | ||
38 | return pte_offset_kernel(pmd, address); | 40 | return pte_offset_kernel(pmd, address); |
39 | } | 41 | } |
@@ -156,11 +158,12 @@ static int __change_page_attr(struct page *page, pgprot_t prot) | |||
156 | struct page *kpte_page; | 158 | struct page *kpte_page; |
157 | unsigned long address; | 159 | unsigned long address; |
158 | pte_t *kpte; | 160 | pte_t *kpte; |
161 | int level; | ||
159 | 162 | ||
160 | BUG_ON(PageHighMem(page)); | 163 | BUG_ON(PageHighMem(page)); |
161 | address = (unsigned long)page_address(page); | 164 | address = (unsigned long)page_address(page); |
162 | 165 | ||
163 | kpte = lookup_address(address); | 166 | kpte = lookup_address(address, &level); |
164 | if (!kpte) | 167 | if (!kpte) |
165 | return -EINVAL; | 168 | return -EINVAL; |
166 | 169 | ||
diff --git a/arch/x86/mm/pageattr_64.c b/arch/x86/mm/pageattr_64.c index 14ab327cde0..59cd066f674 100644 --- a/arch/x86/mm/pageattr_64.c +++ b/arch/x86/mm/pageattr_64.c | |||
@@ -14,7 +14,7 @@ | |||
14 | #include <asm/uaccess.h> | 14 | #include <asm/uaccess.h> |
15 | #include <asm/io.h> | 15 | #include <asm/io.h> |
16 | 16 | ||
17 | pte_t *lookup_address(unsigned long address) | 17 | pte_t *lookup_address(unsigned long address, int *level) |
18 | { | 18 | { |
19 | pgd_t *pgd = pgd_offset_k(address); | 19 | pgd_t *pgd = pgd_offset_k(address); |
20 | pud_t *pud; | 20 | pud_t *pud; |
@@ -29,8 +29,10 @@ pte_t *lookup_address(unsigned long address) | |||
29 | pmd = pmd_offset(pud, address); | 29 | pmd = pmd_offset(pud, address); |
30 | if (!pmd_present(*pmd)) | 30 | if (!pmd_present(*pmd)) |
31 | return NULL; | 31 | return NULL; |
32 | *level = 3; | ||
32 | if (pmd_large(*pmd)) | 33 | if (pmd_large(*pmd)) |
33 | return (pte_t *)pmd; | 34 | return (pte_t *)pmd; |
35 | *level = 4; | ||
34 | 36 | ||
35 | pte = pte_offset_kernel(pmd, address); | 37 | pte = pte_offset_kernel(pmd, address); |
36 | if (pte && !pte_present(*pte)) | 38 | if (pte && !pte_present(*pte)) |
@@ -140,8 +142,9 @@ __change_page_attr(unsigned long address, unsigned long pfn, pgprot_t prot, | |||
140 | struct page *kpte_page; | 142 | struct page *kpte_page; |
141 | pgprot_t ref_prot2; | 143 | pgprot_t ref_prot2; |
142 | pte_t *kpte; | 144 | pte_t *kpte; |
145 | int level; | ||
143 | 146 | ||
144 | kpte = lookup_address(address); | 147 | kpte = lookup_address(address, &level); |
145 | if (!kpte) | 148 | if (!kpte) |
146 | return 0; | 149 | return 0; |
147 | 150 | ||
diff --git a/arch/x86/xen/mmu.c b/arch/x86/xen/mmu.c index e6184735545..45aa771e73a 100644 --- a/arch/x86/xen/mmu.c +++ b/arch/x86/xen/mmu.c | |||
@@ -58,7 +58,8 @@ | |||
58 | 58 | ||
59 | xmaddr_t arbitrary_virt_to_machine(unsigned long address) | 59 | xmaddr_t arbitrary_virt_to_machine(unsigned long address) |
60 | { | 60 | { |
61 | pte_t *pte = lookup_address(address); | 61 | int level; |
62 | pte_t *pte = lookup_address(address, &level); | ||
62 | unsigned offset = address & PAGE_MASK; | 63 | unsigned offset = address & PAGE_MASK; |
63 | 64 | ||
64 | BUG_ON(pte == NULL); | 65 | BUG_ON(pte == NULL); |
@@ -70,8 +71,9 @@ void make_lowmem_page_readonly(void *vaddr) | |||
70 | { | 71 | { |
71 | pte_t *pte, ptev; | 72 | pte_t *pte, ptev; |
72 | unsigned long address = (unsigned long)vaddr; | 73 | unsigned long address = (unsigned long)vaddr; |
74 | int level; | ||
73 | 75 | ||
74 | pte = lookup_address(address); | 76 | pte = lookup_address(address, &level); |
75 | BUG_ON(pte == NULL); | 77 | BUG_ON(pte == NULL); |
76 | 78 | ||
77 | ptev = pte_wrprotect(*pte); | 79 | ptev = pte_wrprotect(*pte); |
@@ -84,8 +86,9 @@ void make_lowmem_page_readwrite(void *vaddr) | |||
84 | { | 86 | { |
85 | pte_t *pte, ptev; | 87 | pte_t *pte, ptev; |
86 | unsigned long address = (unsigned long)vaddr; | 88 | unsigned long address = (unsigned long)vaddr; |
89 | int level; | ||
87 | 90 | ||
88 | pte = lookup_address(address); | 91 | pte = lookup_address(address, &level); |
89 | BUG_ON(pte == NULL); | 92 | BUG_ON(pte == NULL); |
90 | 93 | ||
91 | ptev = pte_mkwrite(*pte); | 94 | ptev = pte_mkwrite(*pte); |
diff --git a/include/asm-x86/pgtable_32.h b/include/asm-x86/pgtable_32.h index eb8cccfa6a4..9381bd37b9b 100644 --- a/include/asm-x86/pgtable_32.h +++ b/include/asm-x86/pgtable_32.h | |||
@@ -182,7 +182,7 @@ static inline void clone_pgd_range(pgd_t *dst, pgd_t *src, int count) | |||
182 | * NOTE: the return type is pte_t but if the pmd is PSE then we return it | 182 | * NOTE: the return type is pte_t but if the pmd is PSE then we return it |
183 | * as a pte too. | 183 | * as a pte too. |
184 | */ | 184 | */ |
185 | extern pte_t *lookup_address(unsigned long address); | 185 | extern pte_t *lookup_address(unsigned long address, int *level); |
186 | 186 | ||
187 | /* | 187 | /* |
188 | * Make a given kernel text page executable/non-executable. | 188 | * Make a given kernel text page executable/non-executable. |
diff --git a/include/asm-x86/pgtable_64.h b/include/asm-x86/pgtable_64.h index 29fdeb8111b..139da50cd51 100644 --- a/include/asm-x86/pgtable_64.h +++ b/include/asm-x86/pgtable_64.h | |||
@@ -240,7 +240,7 @@ extern struct list_head pgd_list; | |||
240 | 240 | ||
241 | extern int kern_addr_valid(unsigned long addr); | 241 | extern int kern_addr_valid(unsigned long addr); |
242 | 242 | ||
243 | pte_t *lookup_address(unsigned long addr); | 243 | pte_t *lookup_address(unsigned long addr, int *level); |
244 | 244 | ||
245 | #define io_remap_pfn_range(vma, vaddr, pfn, size, prot) \ | 245 | #define io_remap_pfn_range(vma, vaddr, pfn, size, prot) \ |
246 | remap_pfn_range(vma, vaddr, pfn, size, prot) | 246 | remap_pfn_range(vma, vaddr, pfn, size, prot) |