aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>2016-04-29 09:26:25 -0400
committerMichael Ellerman <mpe@ellerman.id.au>2016-05-11 07:53:55 -0400
commit484837601d4dd048d1f8d2cedd39934bb3ad6d59 (patch)
treed42a4ce2e1ccfd344743e71ec803cf1e245a2d1f
parent2f5f0dfd1e5f9840560e85ad26cb4a6f06c19e07 (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.h14
-rw-r--r--arch/powerpc/include/asm/hugetlb.h14
-rw-r--r--arch/powerpc/mm/Makefile1
-rw-r--r--arch/powerpc/mm/hugetlbpage-radix.c87
-rw-r--r--arch/powerpc/mm/hugetlbpage.c8
-rw-r--r--arch/powerpc/mm/tlb-radix.c9
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 */
8void radix__flush_hugetlb_page(struct vm_area_struct *vma, unsigned long vmaddr);
9void radix__local_flush_hugetlb_page(struct vm_area_struct *vma, unsigned long vmaddr);
10extern unsigned long
11radix__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 @@
8extern struct kmem_cache *hugepte_cache; 8extern 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}
36static 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
43static 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
37static inline pte_t *hugepd_page(hugepd_t hpd) 51static 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
34obj-y += hugetlbpage.o 34obj-y += hugetlbpage.o
35ifeq ($(CONFIG_HUGETLB_PAGE),y) 35ifeq ($(CONFIG_HUGETLB_PAGE),y)
36obj-$(CONFIG_PPC_STD_MMU_64) += hugetlbpage-hash64.o 36obj-$(CONFIG_PPC_STD_MMU_64) += hugetlbpage-hash64.o
37obj-$(CONFIG_PPC_RADIX_MMU) += hugetlbpage-radix.o
37obj-$(CONFIG_PPC_BOOK3E_MMU) += hugetlbpage-book3e.o 38obj-$(CONFIG_PPC_BOOK3E_MMU) += hugetlbpage-book3e.o
38endif 39endif
39obj-$(CONFIG_TRANSPARENT_HUGEPAGE) += hugepage-hash64.o 40obj-$(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
9void 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
26void 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 */
48unsigned long
49radix__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
142void radix__local_flush_tlb_page(struct vm_area_struct *vma, unsigned long vmaddr) 142void 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
203void radix__flush_tlb_page(struct vm_area_struct *vma, unsigned long vmaddr) 208void 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}