Commit 9043f007 authored by Yinghai Lu's avatar Yinghai Lu Committed by Ingo Molnar

x86, numa, 32-bit: use find_e820_area() to find KVA RAM on node

don't assume we can use RAM near the end of every node.
Esp systems that have few memory and they could have
kva address and kva RAM all below max_low_pfn.
Signed-off-by: default avatarYinghai Lu <yhlu.kernel@gmail.com>
Signed-off-by: default avatarIngo Molnar <mingo@elte.hu>
parent cc1a9d86
...@@ -228,17 +228,21 @@ static unsigned long calculate_numa_remap_pages(void) ...@@ -228,17 +228,21 @@ static unsigned long calculate_numa_remap_pages(void)
{ {
int nid; int nid;
unsigned long size, reserve_pages = 0; unsigned long size, reserve_pages = 0;
unsigned long pfn;
for_each_online_node(nid) { for_each_online_node(nid) {
unsigned old_end_pfn = node_end_pfn[nid]; u64 node_end_target;
u64 node_end_final;
/* /*
* The acpi/srat node info can show hot-add memroy zones * The acpi/srat node info can show hot-add memroy zones
* where memory could be added but not currently present. * where memory could be added but not currently present.
*/ */
printk("node %d pfn: [%lx - %lx]\n",
nid, node_start_pfn[nid], node_end_pfn[nid]);
if (node_start_pfn[nid] > max_pfn) if (node_start_pfn[nid] > max_pfn)
continue; continue;
if (!node_end_pfn[nid])
continue;
if (node_end_pfn[nid] > max_pfn) if (node_end_pfn[nid] > max_pfn)
node_end_pfn[nid] = max_pfn; node_end_pfn[nid] = max_pfn;
...@@ -250,37 +254,40 @@ static unsigned long calculate_numa_remap_pages(void) ...@@ -250,37 +254,40 @@ static unsigned long calculate_numa_remap_pages(void)
/* now the roundup is correct, convert to PAGE_SIZE pages */ /* now the roundup is correct, convert to PAGE_SIZE pages */
size = size * PTRS_PER_PTE; size = size * PTRS_PER_PTE;
/* node_end_target = round_down(node_end_pfn[nid] - size,
* Validate the region we are allocating only contains valid PTRS_PER_PTE);
* pages. node_end_target <<= PAGE_SHIFT;
*/ do {
for (pfn = node_end_pfn[nid] - size; node_end_final = find_e820_area(node_end_target,
pfn < node_end_pfn[nid]; pfn++) ((u64)node_end_pfn[nid])<<PAGE_SHIFT,
if (!page_is_ram(pfn)) ((u64)size)<<PAGE_SHIFT,
break; LARGE_PAGE_BYTES);
node_end_target -= LARGE_PAGE_BYTES;
if (pfn != node_end_pfn[nid]) } while (node_end_final == -1ULL &&
size = 0; (node_end_target>>PAGE_SHIFT) > (node_start_pfn[nid]));
if (node_end_final == -1ULL)
panic("Can not get kva ram\n");
printk("Reserving %ld pages of KVA for lmem_map of node %d\n", printk("Reserving %ld pages of KVA for lmem_map of node %d\n",
size, nid); size, nid);
node_remap_size[nid] = size; node_remap_size[nid] = size;
node_remap_offset[nid] = reserve_pages; node_remap_offset[nid] = reserve_pages;
reserve_pages += size; reserve_pages += size;
printk("Shrinking node %d from %ld pages to %ld pages\n", printk("Shrinking node %d from %ld pages to %lld pages\n",
nid, node_end_pfn[nid], node_end_pfn[nid] - size); nid, node_end_pfn[nid], node_end_final>>PAGE_SHIFT);
if (node_end_pfn[nid] & (PTRS_PER_PTE-1)) { /*
/* * prevent kva address below max_low_pfn want it on system
* Align node_end_pfn[] and node_remap_start_pfn[] to * with less memory later.
* pmd boundary. remap_numa_kva will barf otherwise. * layout will be: KVA address , KVA RAM
*/ */
printk("Shrinking node %d further by %ld pages for proper alignment\n", if ((node_end_final>>PAGE_SHIFT) < max_low_pfn)
nid, node_end_pfn[nid] & (PTRS_PER_PTE-1)); reserve_early(node_end_final,
size += node_end_pfn[nid] & (PTRS_PER_PTE-1); node_end_final+(((u64)size)<<PAGE_SHIFT),
} "KVA RAM");
node_end_pfn[nid] -= size; node_end_pfn[nid] = node_end_final>>PAGE_SHIFT;
node_remap_start_pfn[nid] = node_end_pfn[nid]; node_remap_start_pfn[nid] = node_end_pfn[nid];
shrink_active_range(nid, node_end_pfn[nid]); shrink_active_range(nid, node_end_pfn[nid]);
} }
......
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