aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm64/mm/mmu.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/arm64/mm/mmu.c')
-rw-r--r--arch/arm64/mm/mmu.c67
1 files changed, 30 insertions, 37 deletions
diff --git a/arch/arm64/mm/mmu.c b/arch/arm64/mm/mmu.c
index 4a829a210bb6..c43f1dd19489 100644
--- a/arch/arm64/mm/mmu.c
+++ b/arch/arm64/mm/mmu.c
@@ -43,11 +43,6 @@
43struct page *empty_zero_page; 43struct page *empty_zero_page;
44EXPORT_SYMBOL(empty_zero_page); 44EXPORT_SYMBOL(empty_zero_page);
45 45
46pgprot_t pgprot_default;
47EXPORT_SYMBOL(pgprot_default);
48
49static pmdval_t prot_sect_kernel;
50
51struct cachepolicy { 46struct cachepolicy {
52 const char policy[16]; 47 const char policy[16];
53 u64 mair; 48 u64 mair;
@@ -122,33 +117,6 @@ static int __init early_cachepolicy(char *p)
122} 117}
123early_param("cachepolicy", early_cachepolicy); 118early_param("cachepolicy", early_cachepolicy);
124 119
125/*
126 * Adjust the PMD section entries according to the CPU in use.
127 */
128void __init init_mem_pgprot(void)
129{
130 pteval_t default_pgprot;
131 int i;
132
133 default_pgprot = PTE_ATTRINDX(MT_NORMAL);
134 prot_sect_kernel = PMD_TYPE_SECT | PMD_SECT_AF | PMD_ATTRINDX(MT_NORMAL);
135
136#ifdef CONFIG_SMP
137 /*
138 * Mark memory with the "shared" attribute for SMP systems
139 */
140 default_pgprot |= PTE_SHARED;
141 prot_sect_kernel |= PMD_SECT_S;
142#endif
143
144 for (i = 0; i < 16; i++) {
145 unsigned long v = pgprot_val(protection_map[i]);
146 protection_map[i] = __pgprot(v | default_pgprot);
147 }
148
149 pgprot_default = __pgprot(PTE_TYPE_PAGE | PTE_AF | default_pgprot);
150}
151
152pgprot_t phys_mem_access_prot(struct file *file, unsigned long pfn, 120pgprot_t phys_mem_access_prot(struct file *file, unsigned long pfn,
153 unsigned long size, pgprot_t vma_prot) 121 unsigned long size, pgprot_t vma_prot)
154{ 122{
@@ -196,11 +164,10 @@ static void __init alloc_init_pmd(pud_t *pud, unsigned long addr,
196 pgprot_t prot_pte; 164 pgprot_t prot_pte;
197 165
198 if (map_io) { 166 if (map_io) {
199 prot_sect = PMD_TYPE_SECT | PMD_SECT_AF | 167 prot_sect = PROT_SECT_DEVICE_nGnRE;
200 PMD_ATTRINDX(MT_DEVICE_nGnRE);
201 prot_pte = __pgprot(PROT_DEVICE_nGnRE); 168 prot_pte = __pgprot(PROT_DEVICE_nGnRE);
202 } else { 169 } else {
203 prot_sect = prot_sect_kernel; 170 prot_sect = PROT_SECT_NORMAL_EXEC;
204 prot_pte = PAGE_KERNEL_EXEC; 171 prot_pte = PAGE_KERNEL_EXEC;
205 } 172 }
206 173
@@ -242,7 +209,30 @@ static void __init alloc_init_pud(pgd_t *pgd, unsigned long addr,
242 209
243 do { 210 do {
244 next = pud_addr_end(addr, end); 211 next = pud_addr_end(addr, end);
245 alloc_init_pmd(pud, addr, next, phys, map_io); 212
213 /*
214 * For 4K granule only, attempt to put down a 1GB block
215 */
216 if (!map_io && (PAGE_SHIFT == 12) &&
217 ((addr | next | phys) & ~PUD_MASK) == 0) {
218 pud_t old_pud = *pud;
219 set_pud(pud, __pud(phys | PROT_SECT_NORMAL_EXEC));
220
221 /*
222 * If we have an old value for a pud, it will
223 * be pointing to a pmd table that we no longer
224 * need (from swapper_pg_dir).
225 *
226 * Look up the old pmd table and free it.
227 */
228 if (!pud_none(old_pud)) {
229 phys_addr_t table = __pa(pmd_offset(&old_pud, 0));
230 memblock_free(table, PAGE_SIZE);
231 flush_tlb_all();
232 }
233 } else {
234 alloc_init_pmd(pud, addr, next, phys, map_io);
235 }
246 phys += next - addr; 236 phys += next - addr;
247 } while (pud++, addr = next, addr != end); 237 } while (pud++, addr = next, addr != end);
248} 238}
@@ -399,6 +389,9 @@ int kern_addr_valid(unsigned long addr)
399 if (pud_none(*pud)) 389 if (pud_none(*pud))
400 return 0; 390 return 0;
401 391
392 if (pud_sect(*pud))
393 return pfn_valid(pud_pfn(*pud));
394
402 pmd = pmd_offset(pud, addr); 395 pmd = pmd_offset(pud, addr);
403 if (pmd_none(*pmd)) 396 if (pmd_none(*pmd))
404 return 0; 397 return 0;
@@ -446,7 +439,7 @@ int __meminit vmemmap_populate(unsigned long start, unsigned long end, int node)
446 if (!p) 439 if (!p)
447 return -ENOMEM; 440 return -ENOMEM;
448 441
449 set_pmd(pmd, __pmd(__pa(p) | prot_sect_kernel)); 442 set_pmd(pmd, __pmd(__pa(p) | PROT_SECT_NORMAL));
450 } else 443 } else
451 vmemmap_verify((pte_t *)pmd, node, addr, next); 444 vmemmap_verify((pte_t *)pmd, node, addr, next);
452 } while (addr = next, addr != end); 445 } while (addr = next, addr != end);