Commit e3246d8f authored by Joao Martins's avatar Joao Martins Committed by akpm

mm/sparse-vmemmap: add a pgmap argument to section activation

Patch series "sparse-vmemmap: memory savings for compound devmaps (device-dax)", v9.

This series minimizes 'struct page' overhead by pursuing a similar
approach as Muchun Song series "Free some vmemmap pages of hugetlb page"
(now merged since v5.14), but applied to devmap with @vmemmap_shift
(device-dax).  

The vmemmap dedpulication original idea (already used in HugeTLB) is to
reuse/deduplicate tail page vmemmap areas, particular the area which only
describes tail pages.  So a vmemmap page describes 64 struct pages, and
the first page for a given ZONE_DEVICE vmemmap would contain the head page
and 63 tail pages.  The second vmemmap page would contain only tail pages,
and that's what gets reused across the rest of the subsection/section. 
The bigger the page size, the bigger the savings (2M hpage -> save 6
vmemmap pages; 1G hpage -> save 4094 vmemmap pages).  

This is done for PMEM /specifically only/ on device-dax configured
namespaces, not fsdax.  In other words, a devmap with a @vmemmap_shift.

In terms of savings, per 1Tb of memory, the struct page cost would go down
with compound devmap:

* with 2M pages we lose 4G instead of 16G (0.39% instead of 1.5% of
  total memory)

* with 1G pages we lose 40MB instead of 16G (0.0014% instead of 1.5% of
  total memory)

The series is mostly summed up by patch 4, and to summarize what the
series does:

Patches 1 - 3: Minor cleanups in preparation for patch 4.  Move the very
nice docs of hugetlb_vmemmap.c into a Documentation/vm/ entry.

Patch 4: Patch 4 is the one that takes care of the struct page savings
(also referred to here as tail-page/vmemmap deduplication).  Much like
Muchun series, we reuse the second PTE tail page vmemmap areas across a
given @vmemmap_shift On important difference though, is that contrary to
the hugetlbfs series, there's no vmemmap for the area because we are
late-populating it as opposed to remapping a system-ram range.  IOW no
freeing of pages of already initialized vmemmap like the case for
hugetlbfs, which greatly simplifies the logic (besides not being
arch-specific).  altmap case unchanged and still goes via the
vmemmap_populate().  Also adjust the newly added docs to the device-dax
case.

[Note that device-dax is still a little behind HugeTLB in terms of
savings.  I have an additional simple patch that reuses the head vmemmap
page too, as a follow-up.  That will double the savings and namespaces
initialization.]

Patch 5: Initialize fewer struct pages depending on the page size with
DRAM backed struct pages -- because fewer pages are unique and most tail
pages (with bigger vmemmap_shift).

    NVDIMM namespace bootstrap improves from ~268-358 ms to
    ~80-110/<1ms on 128G NVDIMMs with 2M and 1G respectivally.  And struct
    page needed capacity will be 3.8x / 1071x smaller for 2M and 1G
    respectivelly.  Tested on x86 with 1.5Tb of pmem (including pinning,
    and RDMA registration/deregistration scalability with 2M MRs)


This patch (of 5):

In support of using compound pages for devmap mappings, plumb the pgmap
down to the vmemmap_populate implementation.  Note that while altmap is
retrievable from pgmap the memory hotplug code passes altmap without
pgmap[*], so both need to be independently plumbed.

So in addition to @altmap, pass @pgmap to sparse section populate
functions namely:

	sparse_add_section
	  section_activate
	    populate_section_memmap
   	      __populate_section_memmap

Passing @pgmap allows __populate_section_memmap() to both fetch the
vmemmap_shift in which memmap metadata is created for and also to let
sparse-vmemmap fetch pgmap ranges to co-relate to a given section and pick
whether to just reuse tail pages from past onlined sections.

While at it, fix the kdoc for @altmap for sparse_add_section().

[*] https://lore.kernel.org/linux-mm/20210319092635.6214-1-osalvador@suse.de/

Link: https://lkml.kernel.org/r/20220420155310.9712-1-joao.m.martins@oracle.com
Link: https://lkml.kernel.org/r/20220420155310.9712-2-joao.m.martins@oracle.comSigned-off-by: default avatarJoao Martins <joao.m.martins@oracle.com>
Reviewed-by: default avatarDan Williams <dan.j.williams@intel.com>
Reviewed-by: default avatarMuchun Song <songmuchun@bytedance.com>
Cc: Vishal Verma <vishal.l.verma@intel.com>
Cc: Matthew Wilcox <willy@infradead.org>
Cc: Jason Gunthorpe <jgg@ziepe.ca>
Cc: Jane Chu <jane.chu@oracle.com>
Cc: Mike Kravetz <mike.kravetz@oracle.com>
Cc: Jonathan Corbet <corbet@lwn.net>
Cc: Christoph Hellwig <hch@lst.de>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
parent 47010c04
...@@ -15,6 +15,7 @@ struct memory_block; ...@@ -15,6 +15,7 @@ struct memory_block;
struct memory_group; struct memory_group;
struct resource; struct resource;
struct vmem_altmap; struct vmem_altmap;
struct dev_pagemap;
#ifdef CONFIG_HAVE_ARCH_NODEDATA_EXTENSION #ifdef CONFIG_HAVE_ARCH_NODEDATA_EXTENSION
/* /*
...@@ -122,6 +123,7 @@ typedef int __bitwise mhp_t; ...@@ -122,6 +123,7 @@ typedef int __bitwise mhp_t;
struct mhp_params { struct mhp_params {
struct vmem_altmap *altmap; struct vmem_altmap *altmap;
pgprot_t pgprot; pgprot_t pgprot;
struct dev_pagemap *pgmap;
}; };
bool mhp_range_allowed(u64 start, u64 size, bool need_mapping); bool mhp_range_allowed(u64 start, u64 size, bool need_mapping);
...@@ -333,7 +335,8 @@ extern void remove_pfn_range_from_zone(struct zone *zone, ...@@ -333,7 +335,8 @@ extern void remove_pfn_range_from_zone(struct zone *zone,
unsigned long nr_pages); unsigned long nr_pages);
extern bool is_memblock_offlined(struct memory_block *mem); extern bool is_memblock_offlined(struct memory_block *mem);
extern int sparse_add_section(int nid, unsigned long pfn, extern int sparse_add_section(int nid, unsigned long pfn,
unsigned long nr_pages, struct vmem_altmap *altmap); unsigned long nr_pages, struct vmem_altmap *altmap,
struct dev_pagemap *pgmap);
extern void sparse_remove_section(struct mem_section *ms, extern void sparse_remove_section(struct mem_section *ms,
unsigned long pfn, unsigned long nr_pages, unsigned long pfn, unsigned long nr_pages,
unsigned long map_offset, struct vmem_altmap *altmap); unsigned long map_offset, struct vmem_altmap *altmap);
......
...@@ -3154,7 +3154,8 @@ int vmemmap_remap_alloc(unsigned long start, unsigned long end, ...@@ -3154,7 +3154,8 @@ int vmemmap_remap_alloc(unsigned long start, unsigned long end,
void *sparse_buffer_alloc(unsigned long size); void *sparse_buffer_alloc(unsigned long size);
struct page * __populate_section_memmap(unsigned long pfn, struct page * __populate_section_memmap(unsigned long pfn,
unsigned long nr_pages, int nid, struct vmem_altmap *altmap); unsigned long nr_pages, int nid, struct vmem_altmap *altmap,
struct dev_pagemap *pgmap);
pgd_t *vmemmap_pgd_populate(unsigned long addr, int node); pgd_t *vmemmap_pgd_populate(unsigned long addr, int node);
p4d_t *vmemmap_p4d_populate(pgd_t *pgd, unsigned long addr, int node); p4d_t *vmemmap_p4d_populate(pgd_t *pgd, unsigned long addr, int node);
pud_t *vmemmap_pud_populate(p4d_t *p4d, unsigned long addr, int node); pud_t *vmemmap_pud_populate(p4d_t *p4d, unsigned long addr, int node);
......
...@@ -328,7 +328,8 @@ int __ref __add_pages(int nid, unsigned long pfn, unsigned long nr_pages, ...@@ -328,7 +328,8 @@ int __ref __add_pages(int nid, unsigned long pfn, unsigned long nr_pages,
/* Select all remaining pages up to the next section boundary */ /* Select all remaining pages up to the next section boundary */
cur_nr_pages = min(end_pfn - pfn, cur_nr_pages = min(end_pfn - pfn,
SECTION_ALIGN_UP(pfn + 1) - pfn); SECTION_ALIGN_UP(pfn + 1) - pfn);
err = sparse_add_section(nid, pfn, cur_nr_pages, altmap); err = sparse_add_section(nid, pfn, cur_nr_pages, altmap,
params->pgmap);
if (err) if (err)
break; break;
cond_resched(); cond_resched();
......
...@@ -641,7 +641,8 @@ int __meminit vmemmap_populate_basepages(unsigned long start, unsigned long end, ...@@ -641,7 +641,8 @@ int __meminit vmemmap_populate_basepages(unsigned long start, unsigned long end,
} }
struct page * __meminit __populate_section_memmap(unsigned long pfn, struct page * __meminit __populate_section_memmap(unsigned long pfn,
unsigned long nr_pages, int nid, struct vmem_altmap *altmap) unsigned long nr_pages, int nid, struct vmem_altmap *altmap,
struct dev_pagemap *pgmap)
{ {
unsigned long start = (unsigned long) pfn_to_page(pfn); unsigned long start = (unsigned long) pfn_to_page(pfn);
unsigned long end = start + nr_pages * sizeof(struct page); unsigned long end = start + nr_pages * sizeof(struct page);
......
...@@ -427,7 +427,8 @@ static unsigned long __init section_map_size(void) ...@@ -427,7 +427,8 @@ static unsigned long __init section_map_size(void)
} }
struct page __init *__populate_section_memmap(unsigned long pfn, struct page __init *__populate_section_memmap(unsigned long pfn,
unsigned long nr_pages, int nid, struct vmem_altmap *altmap) unsigned long nr_pages, int nid, struct vmem_altmap *altmap,
struct dev_pagemap *pgmap)
{ {
unsigned long size = section_map_size(); unsigned long size = section_map_size();
struct page *map = sparse_buffer_alloc(size); struct page *map = sparse_buffer_alloc(size);
...@@ -524,7 +525,7 @@ static void __init sparse_init_nid(int nid, unsigned long pnum_begin, ...@@ -524,7 +525,7 @@ static void __init sparse_init_nid(int nid, unsigned long pnum_begin,
break; break;
map = __populate_section_memmap(pfn, PAGES_PER_SECTION, map = __populate_section_memmap(pfn, PAGES_PER_SECTION,
nid, NULL); nid, NULL, NULL);
if (!map) { if (!map) {
pr_err("%s: node[%d] memory map backing failed. Some memory will not be available.", pr_err("%s: node[%d] memory map backing failed. Some memory will not be available.",
__func__, nid); __func__, nid);
...@@ -629,9 +630,10 @@ void offline_mem_sections(unsigned long start_pfn, unsigned long end_pfn) ...@@ -629,9 +630,10 @@ void offline_mem_sections(unsigned long start_pfn, unsigned long end_pfn)
#ifdef CONFIG_SPARSEMEM_VMEMMAP #ifdef CONFIG_SPARSEMEM_VMEMMAP
static struct page * __meminit populate_section_memmap(unsigned long pfn, static struct page * __meminit populate_section_memmap(unsigned long pfn,
unsigned long nr_pages, int nid, struct vmem_altmap *altmap) unsigned long nr_pages, int nid, struct vmem_altmap *altmap,
struct dev_pagemap *pgmap)
{ {
return __populate_section_memmap(pfn, nr_pages, nid, altmap); return __populate_section_memmap(pfn, nr_pages, nid, altmap, pgmap);
} }
static void depopulate_section_memmap(unsigned long pfn, unsigned long nr_pages, static void depopulate_section_memmap(unsigned long pfn, unsigned long nr_pages,
...@@ -700,7 +702,8 @@ static int fill_subsection_map(unsigned long pfn, unsigned long nr_pages) ...@@ -700,7 +702,8 @@ static int fill_subsection_map(unsigned long pfn, unsigned long nr_pages)
} }
#else #else
struct page * __meminit populate_section_memmap(unsigned long pfn, struct page * __meminit populate_section_memmap(unsigned long pfn,
unsigned long nr_pages, int nid, struct vmem_altmap *altmap) unsigned long nr_pages, int nid, struct vmem_altmap *altmap,
struct dev_pagemap *pgmap)
{ {
return kvmalloc_node(array_size(sizeof(struct page), return kvmalloc_node(array_size(sizeof(struct page),
PAGES_PER_SECTION), GFP_KERNEL, nid); PAGES_PER_SECTION), GFP_KERNEL, nid);
...@@ -823,7 +826,8 @@ static void section_deactivate(unsigned long pfn, unsigned long nr_pages, ...@@ -823,7 +826,8 @@ static void section_deactivate(unsigned long pfn, unsigned long nr_pages,
} }
static struct page * __meminit section_activate(int nid, unsigned long pfn, static struct page * __meminit section_activate(int nid, unsigned long pfn,
unsigned long nr_pages, struct vmem_altmap *altmap) unsigned long nr_pages, struct vmem_altmap *altmap,
struct dev_pagemap *pgmap)
{ {
struct mem_section *ms = __pfn_to_section(pfn); struct mem_section *ms = __pfn_to_section(pfn);
struct mem_section_usage *usage = NULL; struct mem_section_usage *usage = NULL;
...@@ -855,7 +859,7 @@ static struct page * __meminit section_activate(int nid, unsigned long pfn, ...@@ -855,7 +859,7 @@ static struct page * __meminit section_activate(int nid, unsigned long pfn,
if (nr_pages < PAGES_PER_SECTION && early_section(ms)) if (nr_pages < PAGES_PER_SECTION && early_section(ms))
return pfn_to_page(pfn); return pfn_to_page(pfn);
memmap = populate_section_memmap(pfn, nr_pages, nid, altmap); memmap = populate_section_memmap(pfn, nr_pages, nid, altmap, pgmap);
if (!memmap) { if (!memmap) {
section_deactivate(pfn, nr_pages, altmap); section_deactivate(pfn, nr_pages, altmap);
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
...@@ -869,7 +873,8 @@ static struct page * __meminit section_activate(int nid, unsigned long pfn, ...@@ -869,7 +873,8 @@ static struct page * __meminit section_activate(int nid, unsigned long pfn,
* @nid: The node to add section on * @nid: The node to add section on
* @start_pfn: start pfn of the memory range * @start_pfn: start pfn of the memory range
* @nr_pages: number of pfns to add in the section * @nr_pages: number of pfns to add in the section
* @altmap: device page map * @altmap: alternate pfns to allocate the memmap backing store
* @pgmap: alternate compound page geometry for devmap mappings
* *
* This is only intended for hotplug. * This is only intended for hotplug.
* *
...@@ -883,7 +888,8 @@ static struct page * __meminit section_activate(int nid, unsigned long pfn, ...@@ -883,7 +888,8 @@ static struct page * __meminit section_activate(int nid, unsigned long pfn,
* * -ENOMEM - Out of memory. * * -ENOMEM - Out of memory.
*/ */
int __meminit sparse_add_section(int nid, unsigned long start_pfn, int __meminit sparse_add_section(int nid, unsigned long start_pfn,
unsigned long nr_pages, struct vmem_altmap *altmap) unsigned long nr_pages, struct vmem_altmap *altmap,
struct dev_pagemap *pgmap)
{ {
unsigned long section_nr = pfn_to_section_nr(start_pfn); unsigned long section_nr = pfn_to_section_nr(start_pfn);
struct mem_section *ms; struct mem_section *ms;
...@@ -894,7 +900,7 @@ int __meminit sparse_add_section(int nid, unsigned long start_pfn, ...@@ -894,7 +900,7 @@ int __meminit sparse_add_section(int nid, unsigned long start_pfn,
if (ret < 0) if (ret < 0)
return ret; return ret;
memmap = section_activate(nid, start_pfn, nr_pages, altmap); memmap = section_activate(nid, start_pfn, nr_pages, altmap, pgmap);
if (IS_ERR(memmap)) if (IS_ERR(memmap))
return PTR_ERR(memmap); return PTR_ERR(memmap);
......
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