Commit ae94da89 authored by Mike Kravetz's avatar Mike Kravetz Committed by Linus Torvalds

hugetlbfs: add arch_hugetlb_valid_size

Patch series "Clean up hugetlb boot command line processing", v4.

Longpeng(Mike) reported a weird message from hugetlb command line
processing and proposed a solution [1].  While the proposed patch does
address the specific issue, there are other related issues in command line
processing.  As hugetlbfs evolved, updates to command line processing have
been made to meet immediate needs and not necessarily in a coordinated
manner.  The result is that some processing is done in arch specific code,
some is done in arch independent code and coordination is problematic.
Semantics can vary between architectures.

The patch series does the following:
- Define arch specific arch_hugetlb_valid_size routine used to validate
  passed huge page sizes.
- Move hugepagesz= command line parsing out of arch specific code and into
  an arch independent routine.
- Clean up command line processing to follow desired semantics and
  document those semantics.

[1] https://lore.kernel.org/linux-mm/20200305033014.1152-1-longpeng2@huawei.com

This patch (of 3):

The architecture independent routine hugetlb_default_setup sets up the
default huge pages size.  It has no way to verify if the passed value is
valid, so it accepts it and attempts to validate at a later time.  This
requires undocumented cooperation between the arch specific and arch
independent code.

For architectures that support more than one huge page size, provide a
routine arch_hugetlb_valid_size to validate a huge page size.
hugetlb_default_setup can use this to validate passed values.

arch_hugetlb_valid_size will also be used in a subsequent patch to move
processing of the "hugepagesz=" in arch specific code to a common routine
in arch independent code.
Signed-off-by: default avatarMike Kravetz <mike.kravetz@oracle.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Acked-by: Gerald Schaefer <gerald.schaefer@de.ibm.com>	[s390]
Acked-by: default avatarWill Deacon <will@kernel.org>
Cc: Catalin Marinas <catalin.marinas@arm.com>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Paul Walmsley <paul.walmsley@sifive.com>
Cc: Palmer Dabbelt <palmer@dabbelt.com>
Cc: Albert Ou <aou@eecs.berkeley.edu>
Cc: Heiko Carstens <heiko.carstens@de.ibm.com>
Cc: Vasily Gorbik <gor@linux.ibm.com>
Cc: Christian Borntraeger <borntraeger@de.ibm.com>
Cc: David S. Miller <davem@davemloft.net>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Dave Hansen <dave.hansen@linux.intel.com>
Cc: Jonathan Corbet <corbet@lwn.net>
Cc: Longpeng <longpeng2@huawei.com>
Cc: Christophe Leroy <christophe.leroy@c-s.fr>
Cc: Randy Dunlap <rdunlap@infradead.org>
Cc: Mina Almasry <almasrymina@google.com>
Cc: Peter Xu <peterx@redhat.com>
Cc: Nitesh Narayan Lal <nitesh@redhat.com>
Cc: Anders Roxell <anders.roxell@linaro.org>
Cc: "Aneesh Kumar K.V" <aneesh.kumar@linux.ibm.com>
Cc: Qian Cai <cai@lca.pw>
Cc: Stephen Rothwell <sfr@canb.auug.org.au>
Link: http://lkml.kernel.org/r/20200428205614.246260-1-mike.kravetz@oracle.com
Link: http://lkml.kernel.org/r/20200428205614.246260-2-mike.kravetz@oracle.com
Link: http://lkml.kernel.org/r/20200417185049.275845-1-mike.kravetz@oracle.com
Link: http://lkml.kernel.org/r/20200417185049.275845-2-mike.kravetz@oracle.comSigned-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 71a2c112
...@@ -464,17 +464,26 @@ static int __init hugetlbpage_init(void) ...@@ -464,17 +464,26 @@ static int __init hugetlbpage_init(void)
} }
arch_initcall(hugetlbpage_init); arch_initcall(hugetlbpage_init);
static __init int setup_hugepagesz(char *opt) bool __init arch_hugetlb_valid_size(unsigned long size)
{ {
unsigned long ps = memparse(opt, &opt); switch (size) {
switch (ps) {
#ifdef CONFIG_ARM64_4K_PAGES #ifdef CONFIG_ARM64_4K_PAGES
case PUD_SIZE: case PUD_SIZE:
#endif #endif
case CONT_PMD_SIZE: case CONT_PMD_SIZE:
case PMD_SIZE: case PMD_SIZE:
case CONT_PTE_SIZE: case CONT_PTE_SIZE:
return true;
}
return false;
}
static __init int setup_hugepagesz(char *opt)
{
unsigned long ps = memparse(opt, &opt);
if (arch_hugetlb_valid_size(ps)) {
add_huge_page_size(ps); add_huge_page_size(ps);
return 1; return 1;
} }
......
...@@ -558,7 +558,7 @@ unsigned long vma_mmu_pagesize(struct vm_area_struct *vma) ...@@ -558,7 +558,7 @@ unsigned long vma_mmu_pagesize(struct vm_area_struct *vma)
return vma_kernel_pagesize(vma); return vma_kernel_pagesize(vma);
} }
static int __init add_huge_page_size(unsigned long long size) bool __init arch_hugetlb_valid_size(unsigned long size)
{ {
int shift = __ffs(size); int shift = __ffs(size);
int mmu_psize; int mmu_psize;
...@@ -566,20 +566,26 @@ static int __init add_huge_page_size(unsigned long long size) ...@@ -566,20 +566,26 @@ static int __init add_huge_page_size(unsigned long long size)
/* Check that it is a page size supported by the hardware and /* Check that it is a page size supported by the hardware and
* that it fits within pagetable and slice limits. */ * that it fits within pagetable and slice limits. */
if (size <= PAGE_SIZE || !is_power_of_2(size)) if (size <= PAGE_SIZE || !is_power_of_2(size))
return -EINVAL; return false;
mmu_psize = check_and_get_huge_psize(shift); mmu_psize = check_and_get_huge_psize(shift);
if (mmu_psize < 0) if (mmu_psize < 0)
return -EINVAL; return false;
BUG_ON(mmu_psize_defs[mmu_psize].shift != shift); BUG_ON(mmu_psize_defs[mmu_psize].shift != shift);
/* Return if huge page size has already been setup */ return true;
if (size_to_hstate(size)) }
return 0;
hugetlb_add_hstate(shift - PAGE_SHIFT); static int __init add_huge_page_size(unsigned long long size)
{
int shift = __ffs(size);
if (!arch_hugetlb_valid_size((unsigned long)size))
return -EINVAL;
if (!size_to_hstate(size))
hugetlb_add_hstate(shift - PAGE_SHIFT);
return 0; return 0;
} }
......
...@@ -12,21 +12,29 @@ int pmd_huge(pmd_t pmd) ...@@ -12,21 +12,29 @@ int pmd_huge(pmd_t pmd)
return pmd_leaf(pmd); return pmd_leaf(pmd);
} }
bool __init arch_hugetlb_valid_size(unsigned long size)
{
if (size == HPAGE_SIZE)
return true;
else if (IS_ENABLED(CONFIG_64BIT) && size == PUD_SIZE)
return true;
else
return false;
}
static __init int setup_hugepagesz(char *opt) static __init int setup_hugepagesz(char *opt)
{ {
unsigned long ps = memparse(opt, &opt); unsigned long ps = memparse(opt, &opt);
if (ps == HPAGE_SIZE) { if (arch_hugetlb_valid_size(ps)) {
hugetlb_add_hstate(HPAGE_SHIFT - PAGE_SHIFT); hugetlb_add_hstate(ilog2(ps) - PAGE_SHIFT);
} else if (IS_ENABLED(CONFIG_64BIT) && ps == PUD_SIZE) { return 1;
hugetlb_add_hstate(PUD_SHIFT - PAGE_SHIFT); }
} else {
hugetlb_bad_size(); hugetlb_bad_size();
pr_err("hugepagesz: Unsupported page size %lu M\n", ps >> 20); pr_err("hugepagesz: Unsupported page size %lu M\n", ps >> 20);
return 0; return 0;
}
return 1;
} }
__setup("hugepagesz=", setup_hugepagesz); __setup("hugepagesz=", setup_hugepagesz);
......
...@@ -254,16 +254,24 @@ follow_huge_pud(struct mm_struct *mm, unsigned long address, ...@@ -254,16 +254,24 @@ follow_huge_pud(struct mm_struct *mm, unsigned long address,
return pud_page(*pud) + ((address & ~PUD_MASK) >> PAGE_SHIFT); return pud_page(*pud) + ((address & ~PUD_MASK) >> PAGE_SHIFT);
} }
bool __init arch_hugetlb_valid_size(unsigned long size)
{
if (MACHINE_HAS_EDAT1 && size == PMD_SIZE)
return true;
else if (MACHINE_HAS_EDAT2 && size == PUD_SIZE)
return true;
else
return false;
}
static __init int setup_hugepagesz(char *opt) static __init int setup_hugepagesz(char *opt)
{ {
unsigned long size; unsigned long size;
char *string = opt; char *string = opt;
size = memparse(opt, &opt); size = memparse(opt, &opt);
if (MACHINE_HAS_EDAT1 && size == PMD_SIZE) { if (arch_hugetlb_valid_size(size)) {
hugetlb_add_hstate(PMD_SHIFT - PAGE_SHIFT); hugetlb_add_hstate(ilog2(size) - PAGE_SHIFT);
} else if (MACHINE_HAS_EDAT2 && size == PUD_SIZE) {
hugetlb_add_hstate(PUD_SHIFT - PAGE_SHIFT);
} else { } else {
hugetlb_bad_size(); hugetlb_bad_size();
pr_err("hugepagesz= specifies an unsupported page size %s\n", pr_err("hugepagesz= specifies an unsupported page size %s\n",
......
...@@ -360,16 +360,11 @@ static void __init pud_huge_patch(void) ...@@ -360,16 +360,11 @@ static void __init pud_huge_patch(void)
__asm__ __volatile__("flush %0" : : "r" (addr)); __asm__ __volatile__("flush %0" : : "r" (addr));
} }
static int __init setup_hugepagesz(char *string) bool __init arch_hugetlb_valid_size(unsigned long size)
{ {
unsigned long long hugepage_size; unsigned int hugepage_shift = ilog2(size);
unsigned int hugepage_shift;
unsigned short hv_pgsz_idx; unsigned short hv_pgsz_idx;
unsigned int hv_pgsz_mask; unsigned int hv_pgsz_mask;
int rc = 0;
hugepage_size = memparse(string, &string);
hugepage_shift = ilog2(hugepage_size);
switch (hugepage_shift) { switch (hugepage_shift) {
case HPAGE_16GB_SHIFT: case HPAGE_16GB_SHIFT:
...@@ -397,7 +392,20 @@ static int __init setup_hugepagesz(char *string) ...@@ -397,7 +392,20 @@ static int __init setup_hugepagesz(char *string)
hv_pgsz_mask = 0; hv_pgsz_mask = 0;
} }
if ((hv_pgsz_mask & cpu_pgsz_mask) == 0U) { if ((hv_pgsz_mask & cpu_pgsz_mask) == 0U)
return false;
return true;
}
static int __init setup_hugepagesz(char *string)
{
unsigned long long hugepage_size;
int rc = 0;
hugepage_size = memparse(string, &string);
if (!arch_hugetlb_valid_size((unsigned long)hugepage_size)) {
hugetlb_bad_size(); hugetlb_bad_size();
pr_err("hugepagesz=%llu not supported by MMU.\n", pr_err("hugepagesz=%llu not supported by MMU.\n",
hugepage_size); hugepage_size);
......
...@@ -181,13 +181,22 @@ hugetlb_get_unmapped_area(struct file *file, unsigned long addr, ...@@ -181,13 +181,22 @@ hugetlb_get_unmapped_area(struct file *file, unsigned long addr,
#endif /* CONFIG_HUGETLB_PAGE */ #endif /* CONFIG_HUGETLB_PAGE */
#ifdef CONFIG_X86_64 #ifdef CONFIG_X86_64
bool __init arch_hugetlb_valid_size(unsigned long size)
{
if (size == PMD_SIZE)
return true;
else if (size == PUD_SIZE && boot_cpu_has(X86_FEATURE_GBPAGES))
return true;
else
return false;
}
static __init int setup_hugepagesz(char *opt) static __init int setup_hugepagesz(char *opt)
{ {
unsigned long ps = memparse(opt, &opt); unsigned long ps = memparse(opt, &opt);
if (ps == PMD_SIZE) {
hugetlb_add_hstate(PMD_SHIFT - PAGE_SHIFT); if (arch_hugetlb_valid_size(ps)) {
} else if (ps == PUD_SIZE && boot_cpu_has(X86_FEATURE_GBPAGES)) { hugetlb_add_hstate(ilog2(ps) - PAGE_SHIFT);
hugetlb_add_hstate(PUD_SHIFT - PAGE_SHIFT);
} else { } else {
hugetlb_bad_size(); hugetlb_bad_size();
printk(KERN_ERR "hugepagesz: Unsupported page size %lu M\n", printk(KERN_ERR "hugepagesz: Unsupported page size %lu M\n",
......
...@@ -521,6 +521,7 @@ int __init alloc_bootmem_huge_page(struct hstate *h); ...@@ -521,6 +521,7 @@ int __init alloc_bootmem_huge_page(struct hstate *h);
void __init hugetlb_bad_size(void); void __init hugetlb_bad_size(void);
void __init hugetlb_add_hstate(unsigned order); void __init hugetlb_add_hstate(unsigned order);
bool __init arch_hugetlb_valid_size(unsigned long size);
struct hstate *size_to_hstate(unsigned long size); struct hstate *size_to_hstate(unsigned long size);
#ifndef HUGE_MAX_HSTATE #ifndef HUGE_MAX_HSTATE
......
...@@ -3256,6 +3256,12 @@ static int __init hugetlb_init(void) ...@@ -3256,6 +3256,12 @@ static int __init hugetlb_init(void)
} }
subsys_initcall(hugetlb_init); subsys_initcall(hugetlb_init);
/* Overwritten by architectures with more huge page sizes */
bool __init __attribute((weak)) arch_hugetlb_valid_size(unsigned long size)
{
return size == HPAGE_SIZE;
}
/* Should be called on processing a hugepagesz=... option */ /* Should be called on processing a hugepagesz=... option */
void __init hugetlb_bad_size(void) void __init hugetlb_bad_size(void)
{ {
...@@ -3331,12 +3337,21 @@ static int __init hugetlb_nrpages_setup(char *s) ...@@ -3331,12 +3337,21 @@ static int __init hugetlb_nrpages_setup(char *s)
} }
__setup("hugepages=", hugetlb_nrpages_setup); __setup("hugepages=", hugetlb_nrpages_setup);
static int __init hugetlb_default_setup(char *s) static int __init default_hugepagesz_setup(char *s)
{ {
default_hstate_size = memparse(s, &s); unsigned long size;
size = (unsigned long)memparse(s, NULL);
if (!arch_hugetlb_valid_size(size)) {
pr_err("HugeTLB: unsupported default_hugepagesz %s\n", s);
return 0;
}
default_hstate_size = size;
return 1; return 1;
} }
__setup("default_hugepagesz=", hugetlb_default_setup); __setup("default_hugepagesz=", default_hugepagesz_setup);
static unsigned int cpuset_mems_nr(unsigned int *array) static unsigned int cpuset_mems_nr(unsigned int *array)
{ {
......
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