mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	ARM64: mm: HugeTLB support.
Add huge page support to ARM64, different huge page sizes are
supported depending on the size of normal pages:
PAGE_SIZE is 4KB:
   2MB - (pmds) these can be allocated at any time.
1024MB - (puds) usually allocated on bootup with the command line
         with something like: hugepagesz=1G hugepages=6
PAGE_SIZE is 64KB:
 512MB - (pmds) usually allocated on bootup via command line.
Signed-off-by: Steve Capper <steve.capper@linaro.org>
Acked-by: Catalin Marinas <catalin.marinas@arm.com>
			
			
This commit is contained in:
		
							parent
							
								
									59911ca432
								
							
						
					
					
						commit
						084bd29810
					
				| @ -180,6 +180,15 @@ config HW_PERF_EVENTS | |||||||
| 	  Enable hardware performance counter support for perf events. If | 	  Enable hardware performance counter support for perf events. If | ||||||
| 	  disabled, perf events will use software events only. | 	  disabled, perf events will use software events only. | ||||||
| 
 | 
 | ||||||
|  | config SYS_SUPPORTS_HUGETLBFS | ||||||
|  | 	def_bool y | ||||||
|  | 
 | ||||||
|  | config ARCH_WANT_GENERAL_HUGETLB | ||||||
|  | 	def_bool y | ||||||
|  | 
 | ||||||
|  | config ARCH_WANT_HUGE_PMD_SHARE | ||||||
|  | 	def_bool y if !ARM64_64K_PAGES | ||||||
|  | 
 | ||||||
| source "mm/Kconfig" | source "mm/Kconfig" | ||||||
| 
 | 
 | ||||||
| endmenu | endmenu | ||||||
|  | |||||||
							
								
								
									
										117
									
								
								arch/arm64/include/asm/hugetlb.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								arch/arm64/include/asm/hugetlb.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,117 @@ | |||||||
|  | /*
 | ||||||
|  |  * arch/arm64/include/asm/hugetlb.h | ||||||
|  |  * | ||||||
|  |  * Copyright (C) 2013 Linaro Ltd. | ||||||
|  |  * | ||||||
|  |  * Based on arch/x86/include/asm/hugetlb.h | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License version 2 as | ||||||
|  |  * published by the Free Software Foundation. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program; if not, write to the Free Software | ||||||
|  |  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #ifndef __ASM_HUGETLB_H | ||||||
|  | #define __ASM_HUGETLB_H | ||||||
|  | 
 | ||||||
|  | #include <asm-generic/hugetlb.h> | ||||||
|  | #include <asm/page.h> | ||||||
|  | 
 | ||||||
|  | static inline pte_t huge_ptep_get(pte_t *ptep) | ||||||
|  | { | ||||||
|  | 	return *ptep; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void set_huge_pte_at(struct mm_struct *mm, unsigned long addr, | ||||||
|  | 				   pte_t *ptep, pte_t pte) | ||||||
|  | { | ||||||
|  | 	set_pte_at(mm, addr, ptep, pte); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void huge_ptep_clear_flush(struct vm_area_struct *vma, | ||||||
|  | 					 unsigned long addr, pte_t *ptep) | ||||||
|  | { | ||||||
|  | 	ptep_clear_flush(vma, addr, ptep); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void huge_ptep_set_wrprotect(struct mm_struct *mm, | ||||||
|  | 					   unsigned long addr, pte_t *ptep) | ||||||
|  | { | ||||||
|  | 	ptep_set_wrprotect(mm, addr, ptep); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline pte_t huge_ptep_get_and_clear(struct mm_struct *mm, | ||||||
|  | 					    unsigned long addr, pte_t *ptep) | ||||||
|  | { | ||||||
|  | 	return ptep_get_and_clear(mm, addr, ptep); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int huge_ptep_set_access_flags(struct vm_area_struct *vma, | ||||||
|  | 					     unsigned long addr, pte_t *ptep, | ||||||
|  | 					     pte_t pte, int dirty) | ||||||
|  | { | ||||||
|  | 	return ptep_set_access_flags(vma, addr, ptep, pte, dirty); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void hugetlb_free_pgd_range(struct mmu_gather *tlb, | ||||||
|  | 					  unsigned long addr, unsigned long end, | ||||||
|  | 					  unsigned long floor, | ||||||
|  | 					  unsigned long ceiling) | ||||||
|  | { | ||||||
|  | 	free_pgd_range(tlb, addr, end, floor, ceiling); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int is_hugepage_only_range(struct mm_struct *mm, | ||||||
|  | 					 unsigned long addr, unsigned long len) | ||||||
|  | { | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int prepare_hugepage_range(struct file *file, | ||||||
|  | 					 unsigned long addr, unsigned long len) | ||||||
|  | { | ||||||
|  | 	struct hstate *h = hstate_file(file); | ||||||
|  | 	if (len & ~huge_page_mask(h)) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	if (addr & ~huge_page_mask(h)) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void hugetlb_prefault_arch_hook(struct mm_struct *mm) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int huge_pte_none(pte_t pte) | ||||||
|  | { | ||||||
|  | 	return pte_none(pte); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline pte_t huge_pte_wrprotect(pte_t pte) | ||||||
|  | { | ||||||
|  | 	return pte_wrprotect(pte); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline int arch_prepare_hugepage(struct page *page) | ||||||
|  | { | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void arch_release_hugepage(struct page *page) | ||||||
|  | { | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void arch_clear_hugepage_flags(struct page *page) | ||||||
|  | { | ||||||
|  | 	clear_bit(PG_dcache_clean, &page->flags); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif /* __ASM_HUGETLB_H */ | ||||||
| @ -25,12 +25,19 @@ | |||||||
| /*
 | /*
 | ||||||
|  * Hardware page table definitions. |  * Hardware page table definitions. | ||||||
|  * |  * | ||||||
|  |  * Level 1 descriptor (PUD). | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #define PUD_TABLE_BIT		(_AT(pgdval_t, 1) << 1) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  * Level 2 descriptor (PMD). |  * Level 2 descriptor (PMD). | ||||||
|  */ |  */ | ||||||
| #define PMD_TYPE_MASK		(_AT(pmdval_t, 3) << 0) | #define PMD_TYPE_MASK		(_AT(pmdval_t, 3) << 0) | ||||||
| #define PMD_TYPE_FAULT		(_AT(pmdval_t, 0) << 0) | #define PMD_TYPE_FAULT		(_AT(pmdval_t, 0) << 0) | ||||||
| #define PMD_TYPE_TABLE		(_AT(pmdval_t, 3) << 0) | #define PMD_TYPE_TABLE		(_AT(pmdval_t, 3) << 0) | ||||||
| #define PMD_TYPE_SECT		(_AT(pmdval_t, 1) << 0) | #define PMD_TYPE_SECT		(_AT(pmdval_t, 1) << 0) | ||||||
|  | #define PMD_TABLE_BIT		(_AT(pmdval_t, 1) << 1) | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Section |  * Section | ||||||
| @ -53,6 +60,7 @@ | |||||||
| #define PTE_TYPE_MASK		(_AT(pteval_t, 3) << 0) | #define PTE_TYPE_MASK		(_AT(pteval_t, 3) << 0) | ||||||
| #define PTE_TYPE_FAULT		(_AT(pteval_t, 0) << 0) | #define PTE_TYPE_FAULT		(_AT(pteval_t, 0) << 0) | ||||||
| #define PTE_TYPE_PAGE		(_AT(pteval_t, 3) << 0) | #define PTE_TYPE_PAGE		(_AT(pteval_t, 3) << 0) | ||||||
|  | #define PTE_TABLE_BIT		(_AT(pteval_t, 1) << 1) | ||||||
| #define PTE_USER		(_AT(pteval_t, 1) << 6)		/* AP[1] */ | #define PTE_USER		(_AT(pteval_t, 1) << 6)		/* AP[1] */ | ||||||
| #define PTE_RDONLY		(_AT(pteval_t, 1) << 7)		/* AP[2] */ | #define PTE_RDONLY		(_AT(pteval_t, 1) << 7)		/* AP[2] */ | ||||||
| #define PTE_SHARED		(_AT(pteval_t, 3) << 8)		/* SH[1:0], inner shareable */ | #define PTE_SHARED		(_AT(pteval_t, 3) << 8)		/* SH[1:0], inner shareable */ | ||||||
|  | |||||||
| @ -173,8 +173,17 @@ static inline void set_pte_at(struct mm_struct *mm, unsigned long addr, | |||||||
| /*
 | /*
 | ||||||
|  * Huge pte definitions. |  * Huge pte definitions. | ||||||
|  */ |  */ | ||||||
| #define pte_huge(pte)		((pte_val(pte) & PTE_TYPE_MASK) == PTE_TYPE_HUGEPAGE) | #define pte_huge(pte)		(!(pte_val(pte) & PTE_TABLE_BIT)) | ||||||
| #define pte_mkhuge(pte)		(__pte((pte_val(pte) & ~PTE_TYPE_MASK) | PTE_TYPE_HUGEPAGE)) | #define pte_mkhuge(pte)		(__pte(pte_val(pte) & ~PTE_TABLE_BIT)) | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Hugetlb definitions. | ||||||
|  |  */ | ||||||
|  | #define HUGE_MAX_HSTATE		2 | ||||||
|  | #define HPAGE_SHIFT		PMD_SHIFT | ||||||
|  | #define HPAGE_SIZE		(_AC(1, UL) << HPAGE_SHIFT) | ||||||
|  | #define HPAGE_MASK		(~(HPAGE_SIZE - 1)) | ||||||
|  | #define HUGETLB_PAGE_ORDER	(HPAGE_SHIFT - PAGE_SHIFT) | ||||||
| 
 | 
 | ||||||
| #define __HAVE_ARCH_PTE_SPECIAL | #define __HAVE_ARCH_PTE_SPECIAL | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -2,3 +2,4 @@ obj-y				:= dma-mapping.o extable.o fault.o init.o \ | |||||||
| 				   cache.o copypage.o flush.o \
 | 				   cache.o copypage.o flush.o \
 | ||||||
| 				   ioremap.o mmap.o pgd.o mmu.o \
 | 				   ioremap.o mmap.o pgd.o mmu.o \
 | ||||||
| 				   context.o tlb.o proc.o | 				   context.o tlb.o proc.o | ||||||
|  | obj-$(CONFIG_HUGETLB_PAGE)	+= hugetlbpage.o | ||||||
|  | |||||||
| @ -364,17 +364,6 @@ static int __kprobes do_translation_fault(unsigned long addr, | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 |  | ||||||
|  * Some section permission faults need to be handled gracefully.  They can |  | ||||||
|  * happen due to a __{get,put}_user during an oops. |  | ||||||
|  */ |  | ||||||
| static int do_sect_fault(unsigned long addr, unsigned int esr, |  | ||||||
| 			 struct pt_regs *regs) |  | ||||||
| { |  | ||||||
| 	do_bad_area(addr, esr, regs); |  | ||||||
| 	return 0; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /*
 | /*
 | ||||||
|  * This abort handler always returns "fault". |  * This abort handler always returns "fault". | ||||||
|  */ |  */ | ||||||
| @ -398,12 +387,12 @@ static struct fault_info { | |||||||
| 	{ do_translation_fault,	SIGSEGV, SEGV_MAPERR,	"level 2 translation fault"	}, | 	{ do_translation_fault,	SIGSEGV, SEGV_MAPERR,	"level 2 translation fault"	}, | ||||||
| 	{ do_page_fault,	SIGSEGV, SEGV_MAPERR,	"level 3 translation fault"	}, | 	{ do_page_fault,	SIGSEGV, SEGV_MAPERR,	"level 3 translation fault"	}, | ||||||
| 	{ do_bad,		SIGBUS,  0,		"reserved access flag fault"	}, | 	{ do_bad,		SIGBUS,  0,		"reserved access flag fault"	}, | ||||||
| 	{ do_bad,		SIGSEGV, SEGV_ACCERR,	"level 1 access flag fault"	}, | 	{ do_page_fault,	SIGSEGV, SEGV_ACCERR,	"level 1 access flag fault"	}, | ||||||
| 	{ do_bad,		SIGSEGV, SEGV_ACCERR,	"level 2 access flag fault"	}, | 	{ do_page_fault,	SIGSEGV, SEGV_ACCERR,	"level 2 access flag fault"	}, | ||||||
| 	{ do_page_fault,	SIGSEGV, SEGV_ACCERR,	"level 3 access flag fault"	}, | 	{ do_page_fault,	SIGSEGV, SEGV_ACCERR,	"level 3 access flag fault"	}, | ||||||
| 	{ do_bad,		SIGBUS,  0,		"reserved permission fault"	}, | 	{ do_bad,		SIGBUS,  0,		"reserved permission fault"	}, | ||||||
| 	{ do_bad,		SIGSEGV, SEGV_ACCERR,	"level 1 permission fault"	}, | 	{ do_page_fault,	SIGSEGV, SEGV_ACCERR,	"level 1 permission fault"	}, | ||||||
| 	{ do_sect_fault,	SIGSEGV, SEGV_ACCERR,	"level 2 permission fault"	}, | 	{ do_page_fault,	SIGSEGV, SEGV_ACCERR,	"level 2 permission fault"	}, | ||||||
| 	{ do_page_fault,	SIGSEGV, SEGV_ACCERR,	"level 3 permission fault"	}, | 	{ do_page_fault,	SIGSEGV, SEGV_ACCERR,	"level 3 permission fault"	}, | ||||||
| 	{ do_bad,		SIGBUS,  0,		"synchronous external abort"	}, | 	{ do_bad,		SIGBUS,  0,		"synchronous external abort"	}, | ||||||
| 	{ do_bad,		SIGBUS,  0,		"asynchronous external abort"	}, | 	{ do_bad,		SIGBUS,  0,		"asynchronous external abort"	}, | ||||||
|  | |||||||
							
								
								
									
										70
									
								
								arch/arm64/mm/hugetlbpage.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								arch/arm64/mm/hugetlbpage.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,70 @@ | |||||||
|  | /*
 | ||||||
|  |  * arch/arm64/mm/hugetlbpage.c | ||||||
|  |  * | ||||||
|  |  * Copyright (C) 2013 Linaro Ltd. | ||||||
|  |  * | ||||||
|  |  * Based on arch/x86/mm/hugetlbpage.c. | ||||||
|  |  * | ||||||
|  |  * This program is free software; you can redistribute it and/or modify | ||||||
|  |  * it under the terms of the GNU General Public License version 2 as | ||||||
|  |  * published by the Free Software Foundation. | ||||||
|  |  * | ||||||
|  |  * This program is distributed in the hope that it will be useful, | ||||||
|  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  |  * GNU General Public License for more details. | ||||||
|  |  * | ||||||
|  |  * You should have received a copy of the GNU General Public License | ||||||
|  |  * along with this program; if not, write to the Free Software | ||||||
|  |  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <linux/init.h> | ||||||
|  | #include <linux/fs.h> | ||||||
|  | #include <linux/mm.h> | ||||||
|  | #include <linux/hugetlb.h> | ||||||
|  | #include <linux/pagemap.h> | ||||||
|  | #include <linux/err.h> | ||||||
|  | #include <linux/sysctl.h> | ||||||
|  | #include <asm/mman.h> | ||||||
|  | #include <asm/tlb.h> | ||||||
|  | #include <asm/tlbflush.h> | ||||||
|  | #include <asm/pgalloc.h> | ||||||
|  | 
 | ||||||
|  | #ifndef CONFIG_ARCH_WANT_HUGE_PMD_SHARE | ||||||
|  | int huge_pmd_unshare(struct mm_struct *mm, unsigned long *addr, pte_t *ptep) | ||||||
|  | { | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | struct page *follow_huge_addr(struct mm_struct *mm, unsigned long address, | ||||||
|  | 			      int write) | ||||||
|  | { | ||||||
|  | 	return ERR_PTR(-EINVAL); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int pmd_huge(pmd_t pmd) | ||||||
|  | { | ||||||
|  | 	return !(pmd_val(pmd) & PMD_TABLE_BIT); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int pud_huge(pud_t pud) | ||||||
|  | { | ||||||
|  | 	return !(pud_val(pud) & PUD_TABLE_BIT); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static __init int setup_hugepagesz(char *opt) | ||||||
|  | { | ||||||
|  | 	unsigned long ps = memparse(opt, &opt); | ||||||
|  | 	if (ps == PMD_SIZE) { | ||||||
|  | 		hugetlb_add_hstate(PMD_SHIFT - PAGE_SHIFT); | ||||||
|  | 	} else if (ps == PUD_SIZE) { | ||||||
|  | 		hugetlb_add_hstate(PUD_SHIFT - PAGE_SHIFT); | ||||||
|  | 	} else { | ||||||
|  | 		pr_err("hugepagesz: Unsupported page size %lu M\n", ps >> 20); | ||||||
|  | 		return 0; | ||||||
|  | 	} | ||||||
|  | 	return 1; | ||||||
|  | } | ||||||
|  | __setup("hugepagesz=", setup_hugepagesz); | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Steve Capper
						Steve Capper