diff options
author | Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com> | 2016-04-29 09:26:25 -0400 |
---|---|---|
committer | Michael Ellerman <mpe@ellerman.id.au> | 2016-05-11 07:53:55 -0400 |
commit | 484837601d4dd048d1f8d2cedd39934bb3ad6d59 (patch) | |
tree | d42a4ce2e1ccfd344743e71ec803cf1e245a2d1f | |
parent | 2f5f0dfd1e5f9840560e85ad26cb4a6f06c19e07 (diff) |
powerpc/mm: Add radix support for hugetlb
Signed-off-by: Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
-rw-r--r-- | arch/powerpc/include/asm/book3s/64/hugetlb-radix.h | 14 | ||||
-rw-r--r-- | arch/powerpc/include/asm/hugetlb.h | 14 | ||||
-rw-r--r-- | arch/powerpc/mm/Makefile | 1 | ||||
-rw-r--r-- | arch/powerpc/mm/hugetlbpage-radix.c | 87 | ||||
-rw-r--r-- | arch/powerpc/mm/hugetlbpage.c | 8 | ||||
-rw-r--r-- | arch/powerpc/mm/tlb-radix.c | 9 |
6 files changed, 132 insertions, 1 deletions
diff --git a/arch/powerpc/include/asm/book3s/64/hugetlb-radix.h b/arch/powerpc/include/asm/book3s/64/hugetlb-radix.h new file mode 100644 index 000000000000..60f47649306f --- /dev/null +++ b/arch/powerpc/include/asm/book3s/64/hugetlb-radix.h | |||
@@ -0,0 +1,14 @@ | |||
1 | #ifndef _ASM_POWERPC_BOOK3S_64_HUGETLB_RADIX_H | ||
2 | #define _ASM_POWERPC_BOOK3S_64_HUGETLB_RADIX_H | ||
3 | /* | ||
4 | * For radix we want generic code to handle hugetlb. But then if we want | ||
5 | * both hash and radix to be enabled together we need to workaround the | ||
6 | * limitations. | ||
7 | */ | ||
8 | void radix__flush_hugetlb_page(struct vm_area_struct *vma, unsigned long vmaddr); | ||
9 | void radix__local_flush_hugetlb_page(struct vm_area_struct *vma, unsigned long vmaddr); | ||
10 | extern unsigned long | ||
11 | radix__hugetlb_get_unmapped_area(struct file *file, unsigned long addr, | ||
12 | unsigned long len, unsigned long pgoff, | ||
13 | unsigned long flags); | ||
14 | #endif | ||
diff --git a/arch/powerpc/include/asm/hugetlb.h b/arch/powerpc/include/asm/hugetlb.h index 42814f0567cc..e2d9f4996e5c 100644 --- a/arch/powerpc/include/asm/hugetlb.h +++ b/arch/powerpc/include/asm/hugetlb.h | |||
@@ -8,6 +8,8 @@ | |||
8 | extern struct kmem_cache *hugepte_cache; | 8 | extern struct kmem_cache *hugepte_cache; |
9 | 9 | ||
10 | #ifdef CONFIG_PPC_BOOK3S_64 | 10 | #ifdef CONFIG_PPC_BOOK3S_64 |
11 | |||
12 | #include <asm/book3s/64/hugetlb-radix.h> | ||
11 | /* | 13 | /* |
12 | * This should work for other subarchs too. But right now we use the | 14 | * This should work for other subarchs too. But right now we use the |
13 | * new format only for 64bit book3s | 15 | * new format only for 64bit book3s |
@@ -31,7 +33,19 @@ static inline unsigned int hugepd_shift(hugepd_t hpd) | |||
31 | { | 33 | { |
32 | return mmu_psize_to_shift(hugepd_mmu_psize(hpd)); | 34 | return mmu_psize_to_shift(hugepd_mmu_psize(hpd)); |
33 | } | 35 | } |
36 | static inline void flush_hugetlb_page(struct vm_area_struct *vma, | ||
37 | unsigned long vmaddr) | ||
38 | { | ||
39 | if (radix_enabled()) | ||
40 | return radix__flush_hugetlb_page(vma, vmaddr); | ||
41 | } | ||
34 | 42 | ||
43 | static inline void __local_flush_hugetlb_page(struct vm_area_struct *vma, | ||
44 | unsigned long vmaddr) | ||
45 | { | ||
46 | if (radix_enabled()) | ||
47 | return radix__local_flush_hugetlb_page(vma, vmaddr); | ||
48 | } | ||
35 | #else | 49 | #else |
36 | 50 | ||
37 | static inline pte_t *hugepd_page(hugepd_t hpd) | 51 | static inline pte_t *hugepd_page(hugepd_t hpd) |
diff --git a/arch/powerpc/mm/Makefile b/arch/powerpc/mm/Makefile index 48aa11ae6a6b..47511dd00599 100644 --- a/arch/powerpc/mm/Makefile +++ b/arch/powerpc/mm/Makefile | |||
@@ -34,6 +34,7 @@ obj-$(CONFIG_PPC_MM_SLICES) += slice.o | |||
34 | obj-y += hugetlbpage.o | 34 | obj-y += hugetlbpage.o |
35 | ifeq ($(CONFIG_HUGETLB_PAGE),y) | 35 | ifeq ($(CONFIG_HUGETLB_PAGE),y) |
36 | obj-$(CONFIG_PPC_STD_MMU_64) += hugetlbpage-hash64.o | 36 | obj-$(CONFIG_PPC_STD_MMU_64) += hugetlbpage-hash64.o |
37 | obj-$(CONFIG_PPC_RADIX_MMU) += hugetlbpage-radix.o | ||
37 | obj-$(CONFIG_PPC_BOOK3E_MMU) += hugetlbpage-book3e.o | 38 | obj-$(CONFIG_PPC_BOOK3E_MMU) += hugetlbpage-book3e.o |
38 | endif | 39 | endif |
39 | obj-$(CONFIG_TRANSPARENT_HUGEPAGE) += hugepage-hash64.o | 40 | obj-$(CONFIG_TRANSPARENT_HUGEPAGE) += hugepage-hash64.o |
diff --git a/arch/powerpc/mm/hugetlbpage-radix.c b/arch/powerpc/mm/hugetlbpage-radix.c new file mode 100644 index 000000000000..1e11559e1aac --- /dev/null +++ b/arch/powerpc/mm/hugetlbpage-radix.c | |||
@@ -0,0 +1,87 @@ | |||
1 | #include <linux/mm.h> | ||
2 | #include <linux/hugetlb.h> | ||
3 | #include <asm/pgtable.h> | ||
4 | #include <asm/pgalloc.h> | ||
5 | #include <asm/cacheflush.h> | ||
6 | #include <asm/machdep.h> | ||
7 | #include <asm/mman.h> | ||
8 | |||
9 | void radix__flush_hugetlb_page(struct vm_area_struct *vma, unsigned long vmaddr) | ||
10 | { | ||
11 | unsigned long ap, shift; | ||
12 | struct hstate *hstate = hstate_file(vma->vm_file); | ||
13 | |||
14 | shift = huge_page_shift(hstate); | ||
15 | if (shift == mmu_psize_defs[MMU_PAGE_2M].shift) | ||
16 | ap = mmu_get_ap(MMU_PAGE_2M); | ||
17 | else if (shift == mmu_psize_defs[MMU_PAGE_1G].shift) | ||
18 | ap = mmu_get_ap(MMU_PAGE_1G); | ||
19 | else { | ||
20 | WARN(1, "Wrong huge page shift\n"); | ||
21 | return ; | ||
22 | } | ||
23 | radix___flush_tlb_page(vma->vm_mm, vmaddr, ap, 0); | ||
24 | } | ||
25 | |||
26 | void radix__local_flush_hugetlb_page(struct vm_area_struct *vma, unsigned long vmaddr) | ||
27 | { | ||
28 | unsigned long ap, shift; | ||
29 | struct hstate *hstate = hstate_file(vma->vm_file); | ||
30 | |||
31 | shift = huge_page_shift(hstate); | ||
32 | if (shift == mmu_psize_defs[MMU_PAGE_2M].shift) | ||
33 | ap = mmu_get_ap(MMU_PAGE_2M); | ||
34 | else if (shift == mmu_psize_defs[MMU_PAGE_1G].shift) | ||
35 | ap = mmu_get_ap(MMU_PAGE_1G); | ||
36 | else { | ||
37 | WARN(1, "Wrong huge page shift\n"); | ||
38 | return ; | ||
39 | } | ||
40 | radix___local_flush_tlb_page(vma->vm_mm, vmaddr, ap, 0); | ||
41 | } | ||
42 | |||
43 | /* | ||
44 | * A vairant of hugetlb_get_unmapped_area doing topdown search | ||
45 | * FIXME!! should we do as x86 does or non hugetlb area does ? | ||
46 | * ie, use topdown or not based on mmap_is_legacy check ? | ||
47 | */ | ||
48 | unsigned long | ||
49 | radix__hugetlb_get_unmapped_area(struct file *file, unsigned long addr, | ||
50 | unsigned long len, unsigned long pgoff, | ||
51 | unsigned long flags) | ||
52 | { | ||
53 | struct mm_struct *mm = current->mm; | ||
54 | struct vm_area_struct *vma; | ||
55 | struct hstate *h = hstate_file(file); | ||
56 | struct vm_unmapped_area_info info; | ||
57 | |||
58 | if (len & ~huge_page_mask(h)) | ||
59 | return -EINVAL; | ||
60 | if (len > TASK_SIZE) | ||
61 | return -ENOMEM; | ||
62 | |||
63 | if (flags & MAP_FIXED) { | ||
64 | if (prepare_hugepage_range(file, addr, len)) | ||
65 | return -EINVAL; | ||
66 | return addr; | ||
67 | } | ||
68 | |||
69 | if (addr) { | ||
70 | addr = ALIGN(addr, huge_page_size(h)); | ||
71 | vma = find_vma(mm, addr); | ||
72 | if (TASK_SIZE - len >= addr && | ||
73 | (!vma || addr + len <= vma->vm_start)) | ||
74 | return addr; | ||
75 | } | ||
76 | /* | ||
77 | * We are always doing an topdown search here. Slice code | ||
78 | * does that too. | ||
79 | */ | ||
80 | info.flags = VM_UNMAPPED_AREA_TOPDOWN; | ||
81 | info.length = len; | ||
82 | info.low_limit = PAGE_SIZE; | ||
83 | info.high_limit = current->mm->mmap_base; | ||
84 | info.align_mask = PAGE_MASK & ~huge_page_mask(h); | ||
85 | info.align_offset = 0; | ||
86 | return vm_unmapped_area(&info); | ||
87 | } | ||
diff --git a/arch/powerpc/mm/hugetlbpage.c b/arch/powerpc/mm/hugetlbpage.c index 7d677ef51033..790c5de3b183 100644 --- a/arch/powerpc/mm/hugetlbpage.c +++ b/arch/powerpc/mm/hugetlbpage.c | |||
@@ -711,6 +711,9 @@ unsigned long hugetlb_get_unmapped_area(struct file *file, unsigned long addr, | |||
711 | struct hstate *hstate = hstate_file(file); | 711 | struct hstate *hstate = hstate_file(file); |
712 | int mmu_psize = shift_to_mmu_psize(huge_page_shift(hstate)); | 712 | int mmu_psize = shift_to_mmu_psize(huge_page_shift(hstate)); |
713 | 713 | ||
714 | if (radix_enabled()) | ||
715 | return radix__hugetlb_get_unmapped_area(file, addr, len, | ||
716 | pgoff, flags); | ||
714 | return slice_get_unmapped_area(addr, len, flags, mmu_psize, 1); | 717 | return slice_get_unmapped_area(addr, len, flags, mmu_psize, 1); |
715 | } | 718 | } |
716 | #endif | 719 | #endif |
@@ -823,7 +826,7 @@ static int __init hugetlbpage_init(void) | |||
823 | { | 826 | { |
824 | int psize; | 827 | int psize; |
825 | 828 | ||
826 | if (!mmu_has_feature(MMU_FTR_16M_PAGE)) | 829 | if (!radix_enabled() && !mmu_has_feature(MMU_FTR_16M_PAGE)) |
827 | return -ENODEV; | 830 | return -ENODEV; |
828 | 831 | ||
829 | for (psize = 0; psize < MMU_PAGE_COUNT; ++psize) { | 832 | for (psize = 0; psize < MMU_PAGE_COUNT; ++psize) { |
@@ -863,6 +866,9 @@ static int __init hugetlbpage_init(void) | |||
863 | HPAGE_SHIFT = mmu_psize_defs[MMU_PAGE_16M].shift; | 866 | HPAGE_SHIFT = mmu_psize_defs[MMU_PAGE_16M].shift; |
864 | else if (mmu_psize_defs[MMU_PAGE_1M].shift) | 867 | else if (mmu_psize_defs[MMU_PAGE_1M].shift) |
865 | HPAGE_SHIFT = mmu_psize_defs[MMU_PAGE_1M].shift; | 868 | HPAGE_SHIFT = mmu_psize_defs[MMU_PAGE_1M].shift; |
869 | else if (mmu_psize_defs[MMU_PAGE_2M].shift) | ||
870 | HPAGE_SHIFT = mmu_psize_defs[MMU_PAGE_2M].shift; | ||
871 | |||
866 | 872 | ||
867 | return 0; | 873 | return 0; |
868 | } | 874 | } |
diff --git a/arch/powerpc/mm/tlb-radix.c b/arch/powerpc/mm/tlb-radix.c index ecfa00f81f1e..0fdaf93a3e09 100644 --- a/arch/powerpc/mm/tlb-radix.c +++ b/arch/powerpc/mm/tlb-radix.c | |||
@@ -141,6 +141,11 @@ void radix___local_flush_tlb_page(struct mm_struct *mm, unsigned long vmaddr, | |||
141 | 141 | ||
142 | void radix__local_flush_tlb_page(struct vm_area_struct *vma, unsigned long vmaddr) | 142 | void radix__local_flush_tlb_page(struct vm_area_struct *vma, unsigned long vmaddr) |
143 | { | 143 | { |
144 | #ifdef CONFIG_HUGETLB_PAGE | ||
145 | /* need the return fix for nohash.c */ | ||
146 | if (vma && is_vm_hugetlb_page(vma)) | ||
147 | return __local_flush_hugetlb_page(vma, vmaddr); | ||
148 | #endif | ||
144 | radix___local_flush_tlb_page(vma ? vma->vm_mm : NULL, vmaddr, | 149 | radix___local_flush_tlb_page(vma ? vma->vm_mm : NULL, vmaddr, |
145 | mmu_get_ap(mmu_virtual_psize), 0); | 150 | mmu_get_ap(mmu_virtual_psize), 0); |
146 | } | 151 | } |
@@ -202,6 +207,10 @@ bail: | |||
202 | 207 | ||
203 | void radix__flush_tlb_page(struct vm_area_struct *vma, unsigned long vmaddr) | 208 | void radix__flush_tlb_page(struct vm_area_struct *vma, unsigned long vmaddr) |
204 | { | 209 | { |
210 | #ifdef CONFIG_HUGETLB_PAGE | ||
211 | if (vma && is_vm_hugetlb_page(vma)) | ||
212 | return flush_hugetlb_page(vma, vmaddr); | ||
213 | #endif | ||
205 | radix___flush_tlb_page(vma ? vma->vm_mm : NULL, vmaddr, | 214 | radix___flush_tlb_page(vma ? vma->vm_mm : NULL, vmaddr, |
206 | mmu_get_ap(mmu_virtual_psize), 0); | 215 | mmu_get_ap(mmu_virtual_psize), 0); |
207 | } | 216 | } |