Commit 7a1be318 authored by Ard Biesheuvel's avatar Ard Biesheuvel Committed by Russell King

ARM: 9012/1: move device tree mapping out of linear region

On ARM, setting up the linear region is tricky, given the constraints
around placement and alignment of the memblocks, and how the kernel
itself as well as the DT are placed in physical memory.

Let's simplify matters a bit, by moving the device tree mapping to the
top of the address space, right between the end of the vmalloc region
and the start of the the fixmap region, and create a read-only mapping
for it that is independent of the size of the linear region, and how it
is organized.

Since this region was formerly used as a guard region, which will now be
populated fully on LPAE builds by this read-only mapping (which will
still be able to function as a guard region for stray writes), bump the
start of the [underutilized] fixmap region by 512 KB as well, to ensure
that there is always a proper guard region here. Doing so still leaves
ample room for the fixmap space, even with NR_CPUS set to its maximum
value of 32.
Tested-by: default avatarLinus Walleij <linus.walleij@linaro.org>
Reviewed-by: default avatarLinus Walleij <linus.walleij@linaro.org>
Reviewed-by: default avatarNicolas Pitre <nico@fluxnic.net>
Signed-off-by: default avatarArd Biesheuvel <ardb@kernel.org>
Signed-off-by: default avatarRussell King <rmk+kernel@armlinux.org.uk>
parent e9a2f8b5
...@@ -45,9 +45,14 @@ fffe8000 fffeffff DTCM mapping area for platforms with ...@@ -45,9 +45,14 @@ fffe8000 fffeffff DTCM mapping area for platforms with
fffe0000 fffe7fff ITCM mapping area for platforms with fffe0000 fffe7fff ITCM mapping area for platforms with
ITCM mounted inside the CPU. ITCM mounted inside the CPU.
ffc00000 ffefffff Fixmap mapping region. Addresses provided ffc80000 ffefffff Fixmap mapping region. Addresses provided
by fix_to_virt() will be located here. by fix_to_virt() will be located here.
ffc00000 ffc7ffff Guard region
ff800000 ffbfffff Permanent, fixed read-only mapping of the
firmware provided DT blob
fee00000 feffffff Mapping of PCI I/O space. This is a static fee00000 feffffff Mapping of PCI I/O space. This is a static
mapping within the vmalloc space. mapping within the vmalloc space.
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
#ifndef _ASM_FIXMAP_H #ifndef _ASM_FIXMAP_H
#define _ASM_FIXMAP_H #define _ASM_FIXMAP_H
#define FIXADDR_START 0xffc00000UL #define FIXADDR_START 0xffc80000UL
#define FIXADDR_END 0xfff00000UL #define FIXADDR_END 0xfff00000UL
#define FIXADDR_TOP (FIXADDR_END - PAGE_SIZE) #define FIXADDR_TOP (FIXADDR_END - PAGE_SIZE)
......
...@@ -67,6 +67,10 @@ ...@@ -67,6 +67,10 @@
*/ */
#define XIP_VIRT_ADDR(physaddr) (MODULES_VADDR + ((physaddr) & 0x000fffff)) #define XIP_VIRT_ADDR(physaddr) (MODULES_VADDR + ((physaddr) & 0x000fffff))
#define FDT_FIXED_BASE UL(0xff800000)
#define FDT_FIXED_SIZE (2 * PMD_SIZE)
#define FDT_VIRT_ADDR(physaddr) ((void *)(FDT_FIXED_BASE | (physaddr) % PMD_SIZE))
#if !defined(CONFIG_SMP) && !defined(CONFIG_ARM_LPAE) #if !defined(CONFIG_SMP) && !defined(CONFIG_ARM_LPAE)
/* /*
* Allow 16MB-aligned ioremap pages * Allow 16MB-aligned ioremap pages
...@@ -107,6 +111,7 @@ extern unsigned long vectors_base; ...@@ -107,6 +111,7 @@ extern unsigned long vectors_base;
#define MODULES_VADDR PAGE_OFFSET #define MODULES_VADDR PAGE_OFFSET
#define XIP_VIRT_ADDR(physaddr) (physaddr) #define XIP_VIRT_ADDR(physaddr) (physaddr)
#define FDT_VIRT_ADDR(physaddr) ((void *)(physaddr))
#endif /* !CONFIG_MMU */ #endif /* !CONFIG_MMU */
......
...@@ -275,9 +275,8 @@ __create_page_tables: ...@@ -275,9 +275,8 @@ __create_page_tables:
*/ */
mov r0, r2, lsr #SECTION_SHIFT mov r0, r2, lsr #SECTION_SHIFT
movs r0, r0, lsl #SECTION_SHIFT movs r0, r0, lsl #SECTION_SHIFT
subne r3, r0, r8 ldrne r3, =FDT_FIXED_BASE >> (SECTION_SHIFT - PMD_ORDER)
addne r3, r3, #PAGE_OFFSET addne r3, r3, r4
addne r3, r4, r3, lsr #(SECTION_SHIFT - PMD_ORDER)
orrne r6, r7, r0 orrne r6, r7, r0
strne r6, [r3], #1 << PMD_ORDER strne r6, [r3], #1 << PMD_ORDER
addne r6, r6, #1 << SECTION_SHIFT addne r6, r6, #1 << SECTION_SHIFT
......
...@@ -18,6 +18,7 @@ ...@@ -18,6 +18,7 @@
#include <linux/of_platform.h> #include <linux/of_platform.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/kexec.h> #include <linux/kexec.h>
#include <linux/libfdt.h>
#include <linux/of_fdt.h> #include <linux/of_fdt.h>
#include <linux/cpu.h> #include <linux/cpu.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
...@@ -89,7 +90,6 @@ unsigned int cacheid __read_mostly; ...@@ -89,7 +90,6 @@ unsigned int cacheid __read_mostly;
EXPORT_SYMBOL(cacheid); EXPORT_SYMBOL(cacheid);
unsigned int __atags_pointer __initdata; unsigned int __atags_pointer __initdata;
void *atags_vaddr __initdata;
unsigned int system_rev; unsigned int system_rev;
EXPORT_SYMBOL(system_rev); EXPORT_SYMBOL(system_rev);
...@@ -1083,13 +1083,18 @@ void __init hyp_mode_check(void) ...@@ -1083,13 +1083,18 @@ void __init hyp_mode_check(void)
void __init setup_arch(char **cmdline_p) void __init setup_arch(char **cmdline_p)
{ {
const struct machine_desc *mdesc = NULL; const struct machine_desc *mdesc = NULL;
void *atags_vaddr = NULL;
if (__atags_pointer) if (__atags_pointer)
atags_vaddr = phys_to_virt(__atags_pointer); atags_vaddr = FDT_VIRT_ADDR(__atags_pointer);
setup_processor(); setup_processor();
if (atags_vaddr) if (atags_vaddr) {
mdesc = setup_machine_fdt(atags_vaddr); mdesc = setup_machine_fdt(atags_vaddr);
if (mdesc)
memblock_reserve(__atags_pointer,
fdt_totalsize(atags_vaddr));
}
if (!mdesc) if (!mdesc)
mdesc = setup_machine_tags(atags_vaddr, __machine_arch_type); mdesc = setup_machine_tags(atags_vaddr, __machine_arch_type);
if (!mdesc) { if (!mdesc) {
......
...@@ -223,7 +223,6 @@ void __init arm_memblock_init(const struct machine_desc *mdesc) ...@@ -223,7 +223,6 @@ void __init arm_memblock_init(const struct machine_desc *mdesc)
if (mdesc->reserve) if (mdesc->reserve)
mdesc->reserve(); mdesc->reserve();
early_init_fdt_reserve_self();
early_init_fdt_scan_reserved_mem(); early_init_fdt_scan_reserved_mem();
/* reserve memory for DMA contiguous allocations */ /* reserve memory for DMA contiguous allocations */
......
...@@ -39,6 +39,8 @@ ...@@ -39,6 +39,8 @@
#include "mm.h" #include "mm.h"
#include "tcm.h" #include "tcm.h"
extern unsigned long __atags_pointer;
/* /*
* empty_zero_page is a special page that is used for * empty_zero_page is a special page that is used for
* zero-initialized data and COW. * zero-initialized data and COW.
...@@ -946,7 +948,7 @@ static void __init create_mapping(struct map_desc *md) ...@@ -946,7 +948,7 @@ static void __init create_mapping(struct map_desc *md)
return; return;
} }
if ((md->type == MT_DEVICE || md->type == MT_ROM) && if (md->type == MT_DEVICE &&
md->virtual >= PAGE_OFFSET && md->virtual < FIXADDR_START && md->virtual >= PAGE_OFFSET && md->virtual < FIXADDR_START &&
(md->virtual < VMALLOC_START || md->virtual >= VMALLOC_END)) { (md->virtual < VMALLOC_START || md->virtual >= VMALLOC_END)) {
pr_warn("BUG: mapping for 0x%08llx at 0x%08lx out of vmalloc space\n", pr_warn("BUG: mapping for 0x%08llx at 0x%08lx out of vmalloc space\n",
...@@ -1333,6 +1335,15 @@ static void __init devicemaps_init(const struct machine_desc *mdesc) ...@@ -1333,6 +1335,15 @@ static void __init devicemaps_init(const struct machine_desc *mdesc)
for (addr = VMALLOC_START; addr < (FIXADDR_TOP & PMD_MASK); addr += PMD_SIZE) for (addr = VMALLOC_START; addr < (FIXADDR_TOP & PMD_MASK); addr += PMD_SIZE)
pmd_clear(pmd_off_k(addr)); pmd_clear(pmd_off_k(addr));
if (__atags_pointer) {
/* create a read-only mapping of the device tree */
map.pfn = __phys_to_pfn(__atags_pointer & SECTION_MASK);
map.virtual = FDT_FIXED_BASE;
map.length = FDT_FIXED_SIZE;
map.type = MT_ROM;
create_mapping(&map);
}
/* /*
* Map the kernel if it is XIP. * Map the kernel if it is XIP.
* It is always first in the modulearea. * It is always first in the modulearea.
...@@ -1489,8 +1500,7 @@ static void __init map_lowmem(void) ...@@ -1489,8 +1500,7 @@ static void __init map_lowmem(void)
} }
#ifdef CONFIG_ARM_PV_FIXUP #ifdef CONFIG_ARM_PV_FIXUP
extern void *atags_vaddr; typedef void pgtables_remap(long long offset, unsigned long pgd);
typedef void pgtables_remap(long long offset, unsigned long pgd, void *bdata);
pgtables_remap lpae_pgtables_remap_asm; pgtables_remap lpae_pgtables_remap_asm;
/* /*
...@@ -1503,7 +1513,6 @@ static void __init early_paging_init(const struct machine_desc *mdesc) ...@@ -1503,7 +1513,6 @@ static void __init early_paging_init(const struct machine_desc *mdesc)
unsigned long pa_pgd; unsigned long pa_pgd;
unsigned int cr, ttbcr; unsigned int cr, ttbcr;
long long offset; long long offset;
void *boot_data;
if (!mdesc->pv_fixup) if (!mdesc->pv_fixup)
return; return;
...@@ -1520,7 +1529,6 @@ static void __init early_paging_init(const struct machine_desc *mdesc) ...@@ -1520,7 +1529,6 @@ static void __init early_paging_init(const struct machine_desc *mdesc)
*/ */
lpae_pgtables_remap = (pgtables_remap *)(unsigned long)__pa(lpae_pgtables_remap_asm); lpae_pgtables_remap = (pgtables_remap *)(unsigned long)__pa(lpae_pgtables_remap_asm);
pa_pgd = __pa(swapper_pg_dir); pa_pgd = __pa(swapper_pg_dir);
boot_data = atags_vaddr;
barrier(); barrier();
pr_info("Switching physical address space to 0x%08llx\n", pr_info("Switching physical address space to 0x%08llx\n",
...@@ -1556,7 +1564,7 @@ static void __init early_paging_init(const struct machine_desc *mdesc) ...@@ -1556,7 +1564,7 @@ static void __init early_paging_init(const struct machine_desc *mdesc)
* needs to be assembly. It's fairly simple, as we're using the * needs to be assembly. It's fairly simple, as we're using the
* temporary tables setup by the initial assembly code. * temporary tables setup by the initial assembly code.
*/ */
lpae_pgtables_remap(offset, pa_pgd, boot_data); lpae_pgtables_remap(offset, pa_pgd);
/* Re-enable the caches and cacheable TLB walks */ /* Re-enable the caches and cacheable TLB walks */
asm volatile("mcr p15, 0, %0, c2, c0, 2" : : "r" (ttbcr)); asm volatile("mcr p15, 0, %0, c2, c0, 2" : : "r" (ttbcr));
......
...@@ -39,8 +39,8 @@ ENTRY(lpae_pgtables_remap_asm) ...@@ -39,8 +39,8 @@ ENTRY(lpae_pgtables_remap_asm)
/* Update level 2 entries for the boot data */ /* Update level 2 entries for the boot data */
add r7, r2, #0x1000 add r7, r2, #0x1000
add r7, r7, r3, lsr #SECTION_SHIFT - L2_ORDER movw r3, #FDT_FIXED_BASE >> (SECTION_SHIFT - L2_ORDER)
bic r7, r7, #(1 << L2_ORDER) - 1 add r7, r7, r3
ldrd r4, r5, [r7] ldrd r4, r5, [r7]
adds r4, r4, r0 adds r4, r4, r0
adc r5, r5, r1 adc r5, r5, r1
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment