The patch titled Account for memmap and optionally the kernel image as holes has been added to the -mm tree. Its filename is account-for-memmap-and-optionally-the-kernel-image-as-holes.patch See http://www.zip.com.au/~akpm/linux/patches/stuff/added-to-mm.txt to find out what to do about this ------------------------------------------------------ Subject: Account for memmap and optionally the kernel image as holes From: Mel Gorman <mel@xxxxxxxxx> The x86_64 code accounted for memmap and some portions of the the DMA zone as holes. This was because those areas would never be reclaimed and accounting for them as memory affects min watermarks. This patch will account for the memmap as a memory hole. Architectures may optionally use set_dma_reserve() if they wish to account for a portion of memory in ZONE_DMA as a hole. Signed-off-by: Mel Gorman <mel@xxxxxxxxx> Cc: Dave Hansen <haveblue@xxxxxxxxxx> Cc: Andy Whitcroft <apw@xxxxxxxxxxxx> Cc: Andi Kleen <ak@xxxxxx> Cc: Benjamin Herrenschmidt <benh@xxxxxxxxxxxxxxxxxxx> Cc: Paul Mackerras <paulus@xxxxxxxxx> Cc: "Keith Mannthey" <kmannth@xxxxxxxxx> Cc: "Luck, Tony" <tony.luck@xxxxxxxxx> Cc: KAMEZAWA Hiroyuki <kamezawa.hiroyu@xxxxxxxxxxxxxx> Cc: Yasunori Goto <y-goto@xxxxxxxxxxxxxx> Signed-off-by: Andrew Morton <akpm@xxxxxxxx> --- arch/x86_64/mm/init.c | 4 + include/linux/mm.h | 1 mm/page_alloc.c | 95 ++++++++++++++++++++++++++++++++++++++++ 3 files changed, 99 insertions(+), 1 deletion(-) diff -puN arch/x86_64/mm/init.c~account-for-memmap-and-optionally-the-kernel-image-as-holes arch/x86_64/mm/init.c --- a/arch/x86_64/mm/init.c~account-for-memmap-and-optionally-the-kernel-image-as-holes +++ a/arch/x86_64/mm/init.c @@ -659,8 +659,10 @@ void __init reserve_bootmem_generic(unsi #else reserve_bootmem(phys, len); #endif - if (phys+len <= MAX_DMA_PFN*PAGE_SIZE) + if (phys+len <= MAX_DMA_PFN*PAGE_SIZE) { dma_reserve += len / PAGE_SIZE; + set_dma_reserve(dma_reserve); + } } int kern_addr_valid(unsigned long addr) diff -puN include/linux/mm.h~account-for-memmap-and-optionally-the-kernel-image-as-holes include/linux/mm.h --- a/include/linux/mm.h~account-for-memmap-and-optionally-the-kernel-image-as-holes +++ a/include/linux/mm.h @@ -979,6 +979,7 @@ extern void sparse_memory_present_with_a extern int early_pfn_to_nid(unsigned long pfn); #endif /* CONFIG_HAVE_ARCH_EARLY_PFN_TO_NID */ #endif /* CONFIG_ARCH_POPULATES_NODE_MAP */ +extern void set_dma_reserve(unsigned long new_dma_reserve); extern void memmap_init_zone(unsigned long, int, unsigned long, unsigned long); extern void setup_per_zone_pages_min(void); extern void mem_init(void); diff -puN mm/page_alloc.c~account-for-memmap-and-optionally-the-kernel-image-as-holes mm/page_alloc.c --- a/mm/page_alloc.c~account-for-memmap-and-optionally-the-kernel-image-as-holes +++ a/mm/page_alloc.c @@ -104,6 +104,7 @@ int min_free_kbytes = 1024; unsigned long __meminitdata nr_kernel_pages; unsigned long __meminitdata nr_all_pages; +static unsigned long __initdata dma_reserve; #ifdef CONFIG_ARCH_POPULATES_NODE_MAP /* @@ -2202,6 +2203,20 @@ unsigned long __init zone_absent_pages_i arch_zone_lowest_possible_pfn[zone_type], arch_zone_highest_possible_pfn[zone_type]); } + +/* Return the zone index a PFN is in */ +int memmap_zone_idx(struct page *lmem_map) +{ + int i; + unsigned long phys_addr = virt_to_phys(lmem_map); + unsigned long pfn = phys_addr >> PAGE_SHIFT; + + for (i = 0; i < MAX_NR_ZONES; i++) + if (pfn < arch_zone_highest_possible_pfn[i]) + break; + + return i; +} #else static inline unsigned long zone_spanned_pages_in_node(int nid, unsigned long zone_type, @@ -2219,6 +2234,11 @@ static inline unsigned long zone_absent_ return zholes_size[zone_type]; } + +static inline int memmap_zone_idx(struct page *lmem_map) +{ + return MAX_NR_ZONES; +} #endif static void __init calculate_node_totalpages(struct pglist_data *pgdat, @@ -2242,6 +2262,58 @@ static void __init calculate_node_totalp realtotalpages); } +#ifdef CONFIG_FLAT_NODE_MEM_MAP +/* Account for mem_map for CONFIG_FLAT_NODE_MEM_MAP */ +unsigned long __meminit account_memmap(struct pglist_data *pgdat, + int zone_index) +{ + unsigned long pages = 0; + if (zone_index == memmap_zone_idx(pgdat->node_mem_map)) { + pages = pgdat->node_spanned_pages; + pages = (pages * sizeof(struct page)) >> PAGE_SHIFT; + printk(KERN_DEBUG "%lu pages used for memmap\n", pages); + } + return pages; +} +#else +/* Account for mem_map for CONFIG_SPARSEMEM */ +unsigned long account_memmap(struct pglist_data *pgdat, int zone_index) +{ + unsigned long pages = 0; + unsigned long memmap_pfn; + struct page *memmap_addr; + int pnum; + unsigned long pgdat_startpfn, pgdat_endpfn; + struct mem_section *section; + + pgdat_startpfn = pgdat->node_start_pfn; + pgdat_endpfn = pgdat_startpfn + pgdat->node_spanned_pages; + + /* Go through valid sections looking for memmap */ + for (pnum = 0; pnum < NR_MEM_SECTIONS; pnum++) { + if (!valid_section_nr(pnum)) + continue; + + section = __nr_to_section(pnum); + if (!section_has_mem_map(section)) + continue; + + memmap_addr = __section_mem_map_addr(section); + memmap_pfn = (unsigned long)memmap_addr >> PAGE_SHIFT; + + if (memmap_pfn < pgdat_startpfn || memmap_pfn >= pgdat_endpfn) + continue; + + if (zone_index == memmap_zone_idx(memmap_addr)) + pages += (PAGES_PER_SECTION * sizeof(struct page)); + } + + pages >>= PAGE_SHIFT; + printk(KERN_DEBUG "%lu pages used for SPARSE memmap\n", pages); + return pages; +} +#endif + /* * Set up the zone data structures: * - mark all pages reserved @@ -2269,6 +2341,14 @@ static void __meminit free_area_init_cor realsize = size - zone_absent_pages_in_node(nid, j, zholes_size); + realsize -= account_memmap(pgdat, j); + /* Account for reserved DMA pages */ + if (j == ZONE_DMA && realsize > dma_reserve) { + realsize -= dma_reserve; + printk(KERN_DEBUG "%lu pages DMA reserved\n", + dma_reserve); + } + if (!is_highmem_idx(j)) nr_kernel_pages += realsize; nr_all_pages += realsize; @@ -2584,6 +2664,21 @@ void __init free_area_init_nodes(unsigne } #endif /* CONFIG_ARCH_POPULATES_NODE_MAP */ +/** + * set_dma_reserve - Account the specified number of pages reserved in ZONE_DMA + * @new_dma_reserve - The number of pages to mark reserved + * + * The per-cpu batchsize and zone watermarks are determined by present_pages. + * In the DMA zone, a significant percentage may be consumed by kernel image + * and other unfreeable allocations which can skew the watermarks badly. This + * function may optionally be used to account for unfreeable pages in + * ZONE_DMA. The effect will be lower watermarks and smaller per-cpu batchsize + */ +void __init set_dma_reserve(unsigned long new_dma_reserve) +{ + dma_reserve = new_dma_reserve; +} + #ifndef CONFIG_NEED_MULTIPLE_NODES static bootmem_data_t contig_bootmem_data; struct pglist_data contig_page_data = { .bdata = &contig_bootmem_data }; _ Patches currently in -mm which might be from mel@xxxxxxxxx are add-__gfp_thisnode-to-avoid-fallback-to-other-nodes-and-ignore.patch add-__gfp_thisnode-to-avoid-fallback-to-other-nodes-and-ignore-fix.patch sys_move_pages-do-not-fall-back-to-other-nodes.patch guarantee-that-the-uncached-allocator-gets-pages-on-the-correct.patch introduce-mechanism-for-registering-active-regions-of-memory.patch have-power-use-add_active_range-and-free_area_init_nodes.patch have-x86-use-add_active_range-and-free_area_init_nodes.patch have-x86_64-use-add_active_range-and-free_area_init_nodes.patch have-ia64-use-add_active_range-and-free_area_init_nodes.patch account-for-memmap-and-optionally-the-kernel-image-as-holes.patch make-swsusp-avoid-memory-holes-and-reserved-memory-regions-on-x86_64.patch - To unsubscribe from this list: send the line "unsubscribe mm-commits" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html