Currently, the setup_vm() does initial page table setup in one-shot very early before enabling MMU. Due to this, the setup_vm() has to map all possible kernel virtual addresses since it does not know size and location of RAM. This means we have kernel mappings for non-existent RAM and any buggy driver (or kernel) code doing out-of-bound access to RAM will not fault and cause underterministic behaviour. Further, the setup_vm() creates PMD mappings (i.e. 2M mappings) for RV64 systems. This means for PAGE_OFFSET=0xffffffe000000000 (i.e. MAXPHYSMEM_128GB=y), the setup_vm() will require 129 pages (i.e. 516 KB) of memory for initial page tables which is never freed. The memory required for initial page tables will further increase if we chose a lower value of PAGE_OFFSET (e.g. 0xffffff0000000000) This patch implements two-staged initial page table setup, as follows: 1. Early (i.e. setup_vm()): This stage maps kernel image and DTB in a early page table (i.e. early_pg_dir). The early_pg_dir will be used only by boot HART so it can be freed as-part of init memory free-up. 2. Final (i.e. setup_vm_final()): This stage maps all possible RAM banks in the final page table (i.e. swapper_pg_dir). The boot HART will start using swapper_pg_dir at the end of setup_vm_final(). All non-boot HARTs directly use the swapper_pg_dir created by boot HART. We have following advantages with this new approach: 1. Kernel mappings for non-existent RAM don't exists anymore. 2. Memory consumed by initial page tables is now indpendent of the chosen PAGE_OFFSET. 3. Memory consumed by initial page tables on RV64 system is 2 pages (i.e. 8 KB) which has significantly reduced and these pages will be freed as-part of the init memory free-up. The patch also provides a foundation for implementing strict kernel mappings where we protect kernel text and rodata using PTE permissions. Suggested-by: Mike Rapoport <rppt@linux.ibm.com> Signed-off-by: Anup Patel <anup.patel@wdc.com> [paul.walmsley@sifive.com: updated to apply; fixed a checkpatch warning] Signed-off-by: Paul Walmsley <paul.walmsley@sifive.com>
		
			
				
	
	
		
			50 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			50 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* SPDX-License-Identifier: GPL-2.0 */
 | |
| /*
 | |
|  * Copyright (C) 2019 Western Digital Corporation or its affiliates.
 | |
|  */
 | |
| 
 | |
| #ifndef _ASM_RISCV_FIXMAP_H
 | |
| #define _ASM_RISCV_FIXMAP_H
 | |
| 
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/sizes.h>
 | |
| #include <asm/page.h>
 | |
| #include <asm/pgtable.h>
 | |
| 
 | |
| /*
 | |
|  * Here we define all the compile-time 'special' virtual addresses.
 | |
|  * The point is to have a constant address at compile time, but to
 | |
|  * set the physical address only in the boot process.
 | |
|  *
 | |
|  * These 'compile-time allocated' memory buffers are page-sized. Use
 | |
|  * set_fixmap(idx,phys) to associate physical memory with fixmap indices.
 | |
|  */
 | |
| enum fixed_addresses {
 | |
| 	FIX_HOLE,
 | |
| #define FIX_FDT_SIZE	SZ_1M
 | |
| 	FIX_FDT_END,
 | |
| 	FIX_FDT = FIX_FDT_END + FIX_FDT_SIZE / PAGE_SIZE - 1,
 | |
| 	FIX_PTE,
 | |
| 	FIX_PMD,
 | |
| 	FIX_EARLYCON_MEM_BASE,
 | |
| 	__end_of_fixed_addresses
 | |
| };
 | |
| 
 | |
| #define FIXADDR_SIZE		(__end_of_fixed_addresses * PAGE_SIZE)
 | |
| #define FIXADDR_TOP		(VMALLOC_START)
 | |
| #define FIXADDR_START		(FIXADDR_TOP - FIXADDR_SIZE)
 | |
| 
 | |
| #define FIXMAP_PAGE_IO		PAGE_KERNEL
 | |
| 
 | |
| #define __early_set_fixmap	__set_fixmap
 | |
| 
 | |
| #define __late_set_fixmap	__set_fixmap
 | |
| #define __late_clear_fixmap(idx) __set_fixmap((idx), 0, FIXMAP_PAGE_CLEAR)
 | |
| 
 | |
| extern void __set_fixmap(enum fixed_addresses idx,
 | |
| 			 phys_addr_t phys, pgprot_t prot);
 | |
| 
 | |
| #include <asm-generic/fixmap.h>
 | |
| 
 | |
| #endif /* _ASM_RISCV_FIXMAP_H */
 |