aboutsummaryrefslogtreecommitdiffstats
path: root/mm/sparse-vmemmap.c
diff options
context:
space:
mode:
authorPavel Tatashin <pasha.tatashin@oracle.com>2018-08-17 18:49:21 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2018-08-17 19:20:32 -0400
commit35fd1eb1e8212c02f6eae24335a9e5b80f9519b4 (patch)
tree623e76f9707e5535a74f74dce659ef6782467fe4 /mm/sparse-vmemmap.c
parent330d6e489a0ab49136561d7f792b1d81bcdbb83c (diff)
mm/sparse: abstract sparse buffer allocations
Patch series "sparse_init rewrite", v6. In sparse_init() we allocate two large buffers to temporary hold usemap and memmap for the whole machine. However, we can avoid doing that if we changed sparse_init() to operated on per-node bases instead of doing it on the whole machine beforehand. As shown by Baoquan http://lkml.kernel.org/r/20180628062857.29658-1-bhe@redhat.com The buffers are large enough to cause machine stop to boot on small memory systems. Another benefit of these changes is that they also obsolete CONFIG_SPARSEMEM_ALLOC_MEM_MAP_TOGETHER. This patch (of 5): When struct pages are allocated for sparse-vmemmap VA layout, we first try to allocate one large buffer, and than if that fails allocate struct pages for each section as we go. The code that allocates buffer is uses global variables and is spread across several call sites. Cleanup the code by introducing three functions to handle the global buffer: sparse_buffer_init() initialize the buffer sparse_buffer_fini() free the remaining part of the buffer sparse_buffer_alloc() alloc from the buffer, and if buffer is empty return NULL Define these functions in sparse.c instead of sparse-vmemmap.c because later we will use them for non-vmemmap sparse allocations as well. [akpm@linux-foundation.org: use PTR_ALIGN()] [akpm@linux-foundation.org: s/BUG_ON/WARN_ON/] Link: http://lkml.kernel.org/r/20180712203730.8703-2-pasha.tatashin@oracle.com Signed-off-by: Pavel Tatashin <pasha.tatashin@oracle.com> Tested-by: Michael Ellerman <mpe@ellerman.id.au> [powerpc] Reviewed-by: Oscar Salvador <osalvador@suse.de> Tested-by: Oscar Salvador <osalvador@suse.de> Cc: Pasha Tatashin <Pavel.Tatashin@microsoft.com> Cc: Steven Sistare <steven.sistare@oracle.com> Cc: Daniel Jordan <daniel.m.jordan@oracle.com> Cc: "Kirill A. Shutemov" <kirill.shutemov@linux.intel.com> Cc: Michal Hocko <mhocko@kernel.org> Cc: Dan Williams <dan.j.williams@intel.com> Cc: Jan Kara <jack@suse.cz> Cc: Jérôme Glisse <jglisse@redhat.com> Cc: Souptick Joarder <jrdr.linux@gmail.com> Cc: Baoquan He <bhe@redhat.com> Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org> Cc: Vlastimil Babka <vbabka@suse.cz> Cc: Wei Yang <richard.weiyang@gmail.com> Cc: Dave Hansen <dave.hansen@intel.com> Cc: David Rientjes <rientjes@google.com> Cc: Ingo Molnar <mingo@kernel.org> Cc: Abdul Haleem <abdhalee@linux.vnet.ibm.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm/sparse-vmemmap.c')
-rw-r--r--mm/sparse-vmemmap.c40
1 files changed, 6 insertions, 34 deletions
diff --git a/mm/sparse-vmemmap.c b/mm/sparse-vmemmap.c
index 95e2c7638a5c..b05c7663c640 100644
--- a/mm/sparse-vmemmap.c
+++ b/mm/sparse-vmemmap.c
@@ -43,12 +43,9 @@ static void * __ref __earlyonly_bootmem_alloc(int node,
43 unsigned long goal) 43 unsigned long goal)
44{ 44{
45 return memblock_virt_alloc_try_nid_raw(size, align, goal, 45 return memblock_virt_alloc_try_nid_raw(size, align, goal,
46 BOOTMEM_ALLOC_ACCESSIBLE, node); 46 BOOTMEM_ALLOC_ACCESSIBLE, node);
47} 47}
48 48
49static void *vmemmap_buf;
50static void *vmemmap_buf_end;
51
52void * __meminit vmemmap_alloc_block(unsigned long size, int node) 49void * __meminit vmemmap_alloc_block(unsigned long size, int node)
53{ 50{
54 /* If the main allocator is up use that, fallback to bootmem. */ 51 /* If the main allocator is up use that, fallback to bootmem. */
@@ -76,18 +73,10 @@ void * __meminit vmemmap_alloc_block(unsigned long size, int node)
76/* need to make sure size is all the same during early stage */ 73/* need to make sure size is all the same during early stage */
77void * __meminit vmemmap_alloc_block_buf(unsigned long size, int node) 74void * __meminit vmemmap_alloc_block_buf(unsigned long size, int node)
78{ 75{
79 void *ptr; 76 void *ptr = sparse_buffer_alloc(size);
80
81 if (!vmemmap_buf)
82 return vmemmap_alloc_block(size, node);
83
84 /* take the from buf */
85 ptr = (void *)ALIGN((unsigned long)vmemmap_buf, size);
86 if (ptr + size > vmemmap_buf_end)
87 return vmemmap_alloc_block(size, node);
88
89 vmemmap_buf = ptr + size;
90 77
78 if (!ptr)
79 ptr = vmemmap_alloc_block(size, node);
91 return ptr; 80 return ptr;
92} 81}
93 82
@@ -279,19 +268,9 @@ void __init sparse_mem_maps_populate_node(struct page **map_map,
279 unsigned long map_count, int nodeid) 268 unsigned long map_count, int nodeid)
280{ 269{
281 unsigned long pnum; 270 unsigned long pnum;
282 unsigned long size = sizeof(struct page) * PAGES_PER_SECTION;
283 void *vmemmap_buf_start;
284 int nr_consumed_maps = 0; 271 int nr_consumed_maps = 0;
285 272
286 size = ALIGN(size, PMD_SIZE); 273 sparse_buffer_init(section_map_size() * map_count, nodeid);
287 vmemmap_buf_start = __earlyonly_bootmem_alloc(nodeid, size * map_count,
288 PMD_SIZE, __pa(MAX_DMA_ADDRESS));
289
290 if (vmemmap_buf_start) {
291 vmemmap_buf = vmemmap_buf_start;
292 vmemmap_buf_end = vmemmap_buf_start + size * map_count;
293 }
294
295 for (pnum = pnum_begin; pnum < pnum_end; pnum++) { 274 for (pnum = pnum_begin; pnum < pnum_end; pnum++) {
296 if (!present_section_nr(pnum)) 275 if (!present_section_nr(pnum))
297 continue; 276 continue;
@@ -303,12 +282,5 @@ void __init sparse_mem_maps_populate_node(struct page **map_map,
303 pr_err("%s: sparsemem memory map backing failed some memory will not be available\n", 282 pr_err("%s: sparsemem memory map backing failed some memory will not be available\n",
304 __func__); 283 __func__);
305 } 284 }
306 285 sparse_buffer_fini();
307 if (vmemmap_buf_start) {
308 /* need to free left buf */
309 memblock_free_early(__pa(vmemmap_buf),
310 vmemmap_buf_end - vmemmap_buf);
311 vmemmap_buf = NULL;
312 vmemmap_buf_end = NULL;
313 }
314} 286}