As mentioned by Liu Jiang and Wu Jiangguo, users could specify DMA, DMA32, and HIGHMEM as movable. In order to ensure the kernel will work correctly, we should exclude these memory ranges out from zone_movable_limit[]. NOTE: Do find_usable_zone_for_movable() to initialize movable_zone so that sanitize_zone_movable_limit() could use it. This is pointed out by Wu Jianguo <wujianguo@xxxxxxxxxx>. Reported-by: Wu Jianguo <wujianguo@xxxxxxxxxx> Signed-off-by: Tang Chen <tangchen@xxxxxxxxxxxxxx> Signed-off-by: Liu Jiang <jiang.liu@xxxxxxxxxx> Reviewed-by: Wen Congyang <wency@xxxxxxxxxxxxxx> Reviewed-by: Lai Jiangshan <laijs@xxxxxxxxxxxxxx> Tested-by: Lin Feng <linfeng@xxxxxxxxxxxxxx> --- mm/page_alloc.c | 55 ++++++++++++++++++++++++++++++++++++++++++++++++++++++- 1 files changed, 54 insertions(+), 1 deletions(-) diff --git a/mm/page_alloc.c b/mm/page_alloc.c index 31d27af..70ed381 100644 --- a/mm/page_alloc.c +++ b/mm/page_alloc.c @@ -4412,6 +4412,58 @@ static unsigned long __meminit zone_absent_pages_in_node(int nid, return __absent_pages_in_range(nid, zone_start_pfn, zone_end_pfn); } +/** + * sanitize_zone_movable_limit - Sanitize the zone_movable_limit array. + * + * zone_movable_limit[] have been initialized when parsing SRAT or + * movablemem_map. This function will try to exclude ZONE_DMA, ZONE_DMA32, + * and HIGHMEM from zone_movable_limit[]. + * + * zone_movable_limit[nid] == 0 means no limit for the node. + * + * Note: Need to be called with movable_zone initialized. + */ +static void __meminit sanitize_zone_movable_limit(void) +{ + int i, nid; + unsigned long start_pfn, end_pfn; + + if (!movablemem_map.nr_map) + return; + + /* Iterate each node id. */ + for_each_node(nid) { + /* If we have no limit for this node, just skip it. */ + if (!zone_movable_limit[nid]) + continue; + +#ifdef CONFIG_ZONE_DMA + /* Skip DMA memory. */ + if (zone_movable_limit[nid] < + arch_zone_highest_possible_pfn[ZONE_DMA]) + zone_movable_limit[nid] = + arch_zone_highest_possible_pfn[ZONE_DMA]; +#endif + +#ifdef CONFIG_ZONE_DMA32 + /* Skip DMA32 memory. */ + if (zone_movable_limit[nid] < + arch_zone_highest_possible_pfn[ZONE_DMA32]) + zone_movable_limit[nid] = + arch_zone_highest_possible_pfn[ZONE_DMA32]; +#endif + +#ifdef CONFIG_HIGHMEM + /* Skip lowmem if ZONE_MOVABLE is highmem. */ + if (zone_movable_is_highmem() && + zone_movable_limit[nid] < + arch_zone_lowest_possible_pfn[ZONE_HIGHMEM]) + zone_movable_limit[nid] = + arch_zone_lowest_possible_pfn[ZONE_HIGHMEM]; +#endif + } +} + #else /* CONFIG_HAVE_MEMBLOCK_NODE_MAP */ static inline unsigned long __meminit zone_spanned_pages_in_node(int nid, unsigned long zone_type, @@ -4826,7 +4878,6 @@ static void __init find_zone_movable_pfns_for_nodes(void) goto out; /* usable_startpfn is the lowest possible pfn ZONE_MOVABLE can be at */ - find_usable_zone_for_movable(); usable_startpfn = arch_zone_lowest_possible_pfn[movable_zone]; restart: @@ -4985,6 +5036,8 @@ void __init free_area_init_nodes(unsigned long *max_zone_pfn) /* Find the PFNs that ZONE_MOVABLE begins at in each node */ memset(zone_movable_pfn, 0, sizeof(zone_movable_pfn)); + find_usable_zone_for_movable(); + sanitize_zone_movable_limit(); find_zone_movable_pfns_for_nodes(); /* Print out the zone ranges */ -- 1.7.1 -- To unsubscribe, send a message with 'unsubscribe linux-mm' in the body to majordomo@xxxxxxxxx. For more info on Linux MM, see: http://www.linux-mm.org/ . Don't email: <a href=mailto:"dont@xxxxxxxxx"> email@xxxxxxxxx </a>