Commit 79c1c594 authored by Christophe Leroy's avatar Christophe Leroy Committed by Linus Torvalds

mm/hugetlb: change parameters of arch_make_huge_pte()

Patch series "Subject: [PATCH v2 0/5] Implement huge VMAP and VMALLOC on powerpc 8xx", v2.

This series implements huge VMAP and VMALLOC on powerpc 8xx.

Powerpc 8xx has 4 page sizes:
- 4k
- 16k
- 512k
- 8M

At the time being, vmalloc and vmap only support huge pages which are
leaf at PMD level.

Here the PMD level is 4M, it doesn't correspond to any supported
page size.

For now, implement use of 16k and 512k pages which is done
at PTE level.

Support of 8M pages will be implemented later, it requires use of
hugepd tables.

To allow this, the architecture provides two functions:
- arch_vmap_pte_range_map_size() which tells vmap_pte_range() what
page size to use. A stub returning PAGE_SIZE is provided when the
architecture doesn't provide this function.
- arch_vmap_pte_supported_shift() which tells __vmalloc_node_range()
what page shift to use for a given area size. A stub returning
PAGE_SHIFT is provided when the architecture doesn't provide this
function.

This patch (of 5):

At the time being, arch_make_huge_pte() has the following prototype:

  pte_t arch_make_huge_pte(pte_t entry, struct vm_area_struct *vma,
			   struct page *page, int writable);

vma is used to get the pages shift or size.
vma is also used on Sparc to get vm_flags.
page is not used.
writable is not used.

In order to use this function without a vma, replace vma by shift and
flags.  Also remove the used parameters.

Link: https://lkml.kernel.org/r/cover.1620795204.git.christophe.leroy@csgroup.eu
Link: https://lkml.kernel.org/r/f4633ac6a7da2f22f31a04a89e0a7026bb78b15b.1620795204.git.christophe.leroy@csgroup.euSigned-off-by: default avatarChristophe Leroy <christophe.leroy@csgroup.eu>
Acked-by: default avatarMike Kravetz <mike.kravetz@oracle.com>
Cc: Nicholas Piggin <npiggin@gmail.com>
Cc: Mike Kravetz <mike.kravetz@oracle.com>
Cc: Mike Rapoport <rppt@kernel.org>
Cc: Michael Ellerman <mpe@ellerman.id.au>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Uladzislau Rezki <uladzislau.rezki@sony.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent babbbdd0
...@@ -23,8 +23,7 @@ static inline void arch_clear_hugepage_flags(struct page *page) ...@@ -23,8 +23,7 @@ static inline void arch_clear_hugepage_flags(struct page *page)
} }
#define arch_clear_hugepage_flags arch_clear_hugepage_flags #define arch_clear_hugepage_flags arch_clear_hugepage_flags
extern pte_t arch_make_huge_pte(pte_t entry, struct vm_area_struct *vma, pte_t arch_make_huge_pte(pte_t entry, unsigned int shift, vm_flags_t flags);
struct page *page, int writable);
#define arch_make_huge_pte arch_make_huge_pte #define arch_make_huge_pte arch_make_huge_pte
#define __HAVE_ARCH_HUGE_SET_HUGE_PTE_AT #define __HAVE_ARCH_HUGE_SET_HUGE_PTE_AT
extern void set_huge_pte_at(struct mm_struct *mm, unsigned long addr, extern void set_huge_pte_at(struct mm_struct *mm, unsigned long addr,
......
...@@ -339,10 +339,9 @@ pte_t *huge_pte_offset(struct mm_struct *mm, ...@@ -339,10 +339,9 @@ pte_t *huge_pte_offset(struct mm_struct *mm,
return NULL; return NULL;
} }
pte_t arch_make_huge_pte(pte_t entry, struct vm_area_struct *vma, pte_t arch_make_huge_pte(pte_t entry, unsigned int shift, vm_flags_t flags)
struct page *page, int writable)
{ {
size_t pagesize = huge_page_size(hstate_vma(vma)); size_t pagesize = 1UL << shift;
if (pagesize == CONT_PTE_SIZE) { if (pagesize == CONT_PTE_SIZE) {
entry = pte_mkcont(entry); entry = pte_mkcont(entry);
......
...@@ -66,10 +66,9 @@ static inline void huge_ptep_set_wrprotect(struct mm_struct *mm, ...@@ -66,10 +66,9 @@ static inline void huge_ptep_set_wrprotect(struct mm_struct *mm,
} }
#ifdef CONFIG_PPC_4K_PAGES #ifdef CONFIG_PPC_4K_PAGES
static inline pte_t arch_make_huge_pte(pte_t entry, struct vm_area_struct *vma, static inline pte_t arch_make_huge_pte(pte_t entry, unsigned int shift, vm_flags_t flags)
struct page *page, int writable)
{ {
size_t size = huge_page_size(hstate_vma(vma)); size_t size = 1UL << shift;
if (size == SZ_16K) if (size == SZ_16K)
return __pte(pte_val(entry) & ~_PAGE_HUGE); return __pte(pte_val(entry) & ~_PAGE_HUGE);
......
...@@ -377,8 +377,7 @@ static inline pgprot_t pgprot_noncached(pgprot_t prot) ...@@ -377,8 +377,7 @@ static inline pgprot_t pgprot_noncached(pgprot_t prot)
#define pgprot_noncached pgprot_noncached #define pgprot_noncached pgprot_noncached
#if defined(CONFIG_HUGETLB_PAGE) || defined(CONFIG_TRANSPARENT_HUGEPAGE) #if defined(CONFIG_HUGETLB_PAGE) || defined(CONFIG_TRANSPARENT_HUGEPAGE)
extern pte_t arch_make_huge_pte(pte_t entry, struct vm_area_struct *vma, pte_t arch_make_huge_pte(pte_t entry, unsigned int shift, vm_flags_t flags);
struct page *page, int writable);
#define arch_make_huge_pte arch_make_huge_pte #define arch_make_huge_pte arch_make_huge_pte
static inline unsigned long __pte_default_huge_mask(void) static inline unsigned long __pte_default_huge_mask(void)
{ {
......
...@@ -177,10 +177,8 @@ static pte_t hugepage_shift_to_tte(pte_t entry, unsigned int shift) ...@@ -177,10 +177,8 @@ static pte_t hugepage_shift_to_tte(pte_t entry, unsigned int shift)
return sun4u_hugepage_shift_to_tte(entry, shift); return sun4u_hugepage_shift_to_tte(entry, shift);
} }
pte_t arch_make_huge_pte(pte_t entry, struct vm_area_struct *vma, pte_t arch_make_huge_pte(pte_t entry, unsigned int shift, vm_flags_t flags)
struct page *page, int writeable)
{ {
unsigned int shift = huge_page_shift(hstate_vma(vma));
pte_t pte; pte_t pte;
pte = hugepage_shift_to_tte(entry, shift); pte = hugepage_shift_to_tte(entry, shift);
...@@ -188,7 +186,7 @@ pte_t arch_make_huge_pte(pte_t entry, struct vm_area_struct *vma, ...@@ -188,7 +186,7 @@ pte_t arch_make_huge_pte(pte_t entry, struct vm_area_struct *vma,
#ifdef CONFIG_SPARC64 #ifdef CONFIG_SPARC64
/* If this vma has ADI enabled on it, turn on TTE.mcd /* If this vma has ADI enabled on it, turn on TTE.mcd
*/ */
if (vma->vm_flags & VM_SPARC_ADI) if (flags & VM_SPARC_ADI)
return pte_mkmcd(pte); return pte_mkmcd(pte);
else else
return pte_mknotmcd(pte); return pte_mknotmcd(pte);
......
...@@ -741,8 +741,8 @@ static inline void arch_clear_hugepage_flags(struct page *page) { } ...@@ -741,8 +741,8 @@ static inline void arch_clear_hugepage_flags(struct page *page) { }
#endif #endif
#ifndef arch_make_huge_pte #ifndef arch_make_huge_pte
static inline pte_t arch_make_huge_pte(pte_t entry, struct vm_area_struct *vma, static inline pte_t arch_make_huge_pte(pte_t entry, unsigned int shift,
struct page *page, int writable) vm_flags_t flags)
{ {
return entry; return entry;
} }
......
...@@ -4060,6 +4060,7 @@ static pte_t make_huge_pte(struct vm_area_struct *vma, struct page *page, ...@@ -4060,6 +4060,7 @@ static pte_t make_huge_pte(struct vm_area_struct *vma, struct page *page,
int writable) int writable)
{ {
pte_t entry; pte_t entry;
unsigned int shift = huge_page_shift(hstate_vma(vma));
if (writable) { if (writable) {
entry = huge_pte_mkwrite(huge_pte_mkdirty(mk_huge_pte(page, entry = huge_pte_mkwrite(huge_pte_mkdirty(mk_huge_pte(page,
...@@ -4070,7 +4071,7 @@ static pte_t make_huge_pte(struct vm_area_struct *vma, struct page *page, ...@@ -4070,7 +4071,7 @@ static pte_t make_huge_pte(struct vm_area_struct *vma, struct page *page,
} }
entry = pte_mkyoung(entry); entry = pte_mkyoung(entry);
entry = pte_mkhuge(entry); entry = pte_mkhuge(entry);
entry = arch_make_huge_pte(entry, vma, page, writable); entry = arch_make_huge_pte(entry, shift, vma->vm_flags);
return entry; return entry;
} }
...@@ -5468,10 +5469,11 @@ unsigned long hugetlb_change_protection(struct vm_area_struct *vma, ...@@ -5468,10 +5469,11 @@ unsigned long hugetlb_change_protection(struct vm_area_struct *vma,
} }
if (!huge_pte_none(pte)) { if (!huge_pte_none(pte)) {
pte_t old_pte; pte_t old_pte;
unsigned int shift = huge_page_shift(hstate_vma(vma));
old_pte = huge_ptep_modify_prot_start(vma, address, ptep); old_pte = huge_ptep_modify_prot_start(vma, address, ptep);
pte = pte_mkhuge(huge_pte_modify(old_pte, newprot)); pte = pte_mkhuge(huge_pte_modify(old_pte, newprot));
pte = arch_make_huge_pte(pte, vma, NULL, 0); pte = arch_make_huge_pte(pte, shift, vma->vm_flags);
huge_ptep_modify_prot_commit(vma, address, ptep, old_pte, pte); huge_ptep_modify_prot_commit(vma, address, ptep, old_pte, pte);
pages++; pages++;
} }
......
...@@ -226,8 +226,10 @@ static bool remove_migration_pte(struct page *page, struct vm_area_struct *vma, ...@@ -226,8 +226,10 @@ static bool remove_migration_pte(struct page *page, struct vm_area_struct *vma,
#ifdef CONFIG_HUGETLB_PAGE #ifdef CONFIG_HUGETLB_PAGE
if (PageHuge(new)) { if (PageHuge(new)) {
unsigned int shift = huge_page_shift(hstate_vma(vma));
pte = pte_mkhuge(pte); pte = pte_mkhuge(pte);
pte = arch_make_huge_pte(pte, vma, new, 0); pte = arch_make_huge_pte(pte, shift, vma->vm_flags);
set_huge_pte_at(vma->vm_mm, pvmw.address, pvmw.pte, pte); set_huge_pte_at(vma->vm_mm, pvmw.address, pvmw.pte, pte);
if (PageAnon(new)) if (PageAnon(new))
hugepage_add_anon_rmap(new, vma, pvmw.address); hugepage_add_anon_rmap(new, vma, pvmw.address);
......
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