aboutsummaryrefslogtreecommitdiffstats
path: root/mm/memory_hotplug.c
diff options
context:
space:
mode:
authorYasuaki Ishimatsu <isimatu.yasuaki@jp.fujitsu.com>2013-02-22 19:33:00 -0500
committerLinus Torvalds <torvalds@linux-foundation.org>2013-02-23 20:50:12 -0500
commit46723bfa540f0a1e494476a1734d03626a0bd1e0 (patch)
treec8d3ef712dd67b45c9334f04edeec4aa981a2e29 /mm/memory_hotplug.c
parent24d335ca3606b610ec69c66a1e42760c96d89470 (diff)
memory-hotplug: implement register_page_bootmem_info_section of sparse-vmemmap
For removing memmap region of sparse-vmemmap which is allocated bootmem, memmap region of sparse-vmemmap needs to be registered by get_page_bootmem(). So the patch searches pages of virtual mapping and registers the pages by get_page_bootmem(). NOTE: register_page_bootmem_memmap() is not implemented for ia64, ppc, s390, and sparc. So introduce CONFIG_HAVE_BOOTMEM_INFO_NODE and revert register_page_bootmem_info_node() when platform doesn't support it. It's implemented by adding a new Kconfig option named CONFIG_HAVE_BOOTMEM_INFO_NODE, which will be automatically selected by memory-hotplug feature fully supported archs(currently only on x86_64). Since we have 2 config options called MEMORY_HOTPLUG and MEMORY_HOTREMOVE used for memory hot-add and hot-remove separately, and codes in function register_page_bootmem_info_node() are only used for collecting infomation for hot-remove, so reside it under MEMORY_HOTREMOVE. Besides page_isolation.c selected by MEMORY_ISOLATION under MEMORY_HOTPLUG is also such case, move it too. [mhocko@suse.cz: put register_page_bootmem_memmap inside CONFIG_MEMORY_HOTPLUG_SPARSE] [linfeng@cn.fujitsu.com: introduce CONFIG_HAVE_BOOTMEM_INFO_NODE and revert register_page_bootmem_info_node()] [mhocko@suse.cz: remove the arch specific functions without any implementation] [linfeng@cn.fujitsu.com: mm/Kconfig: move auto selects from MEMORY_HOTPLUG to MEMORY_HOTREMOVE as needed] [rientjes@google.com: fix defined but not used warning] Signed-off-by: Wen Congyang <wency@cn.fujitsu.com> Signed-off-by: Yasuaki Ishimatsu <isimatu.yasuaki@jp.fujitsu.com> Signed-off-by: Tang Chen <tangchen@cn.fujitsu.com> Reviewed-by: Wu Jianguo <wujianguo@huawei.com> Cc: KOSAKI Motohiro <kosaki.motohiro@jp.fujitsu.com> Cc: Jiang Liu <jiang.liu@huawei.com> Cc: Jianguo Wu <wujianguo@huawei.com> Cc: Kamezawa Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com> Cc: Lai Jiangshan <laijs@cn.fujitsu.com> Cc: Ingo Molnar <mingo@elte.hu> Cc: Thomas Gleixner <tglx@linutronix.de> Cc: "H. Peter Anvin" <hpa@zytor.com> Signed-off-by: Michal Hocko <mhocko@suse.cz> Signed-off-by: Lin Feng <linfeng@cn.fujitsu.com> Signed-off-by: David Rientjes <rientjes@google.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm/memory_hotplug.c')
-rw-r--r--mm/memory_hotplug.c35
1 files changed, 31 insertions, 4 deletions
diff --git a/mm/memory_hotplug.c b/mm/memory_hotplug.c
index 942b43f6d736..6c90d222ec0a 100644
--- a/mm/memory_hotplug.c
+++ b/mm/memory_hotplug.c
@@ -91,9 +91,8 @@ static void release_memory_resource(struct resource *res)
91} 91}
92 92
93#ifdef CONFIG_MEMORY_HOTPLUG_SPARSE 93#ifdef CONFIG_MEMORY_HOTPLUG_SPARSE
94#ifndef CONFIG_SPARSEMEM_VMEMMAP 94void get_page_bootmem(unsigned long info, struct page *page,
95static void get_page_bootmem(unsigned long info, struct page *page, 95 unsigned long type)
96 unsigned long type)
97{ 96{
98 page->lru.next = (struct list_head *) type; 97 page->lru.next = (struct list_head *) type;
99 SetPagePrivate(page); 98 SetPagePrivate(page);
@@ -128,6 +127,8 @@ void __ref put_page_bootmem(struct page *page)
128 127
129} 128}
130 129
130#ifdef CONFIG_HAVE_BOOTMEM_INFO_NODE
131#ifndef CONFIG_SPARSEMEM_VMEMMAP
131static void register_page_bootmem_info_section(unsigned long start_pfn) 132static void register_page_bootmem_info_section(unsigned long start_pfn)
132{ 133{
133 unsigned long *usemap, mapsize, section_nr, i; 134 unsigned long *usemap, mapsize, section_nr, i;
@@ -161,6 +162,32 @@ static void register_page_bootmem_info_section(unsigned long start_pfn)
161 get_page_bootmem(section_nr, page, MIX_SECTION_INFO); 162 get_page_bootmem(section_nr, page, MIX_SECTION_INFO);
162 163
163} 164}
165#else /* CONFIG_SPARSEMEM_VMEMMAP */
166static void register_page_bootmem_info_section(unsigned long start_pfn)
167{
168 unsigned long *usemap, mapsize, section_nr, i;
169 struct mem_section *ms;
170 struct page *page, *memmap;
171
172 if (!pfn_valid(start_pfn))
173 return;
174
175 section_nr = pfn_to_section_nr(start_pfn);
176 ms = __nr_to_section(section_nr);
177
178 memmap = sparse_decode_mem_map(ms->section_mem_map, section_nr);
179
180 register_page_bootmem_memmap(section_nr, memmap, PAGES_PER_SECTION);
181
182 usemap = __nr_to_section(section_nr)->pageblock_flags;
183 page = virt_to_page(usemap);
184
185 mapsize = PAGE_ALIGN(usemap_size()) >> PAGE_SHIFT;
186
187 for (i = 0; i < mapsize; i++, page++)
188 get_page_bootmem(section_nr, page, MIX_SECTION_INFO);
189}
190#endif /* !CONFIG_SPARSEMEM_VMEMMAP */
164 191
165void register_page_bootmem_info_node(struct pglist_data *pgdat) 192void register_page_bootmem_info_node(struct pglist_data *pgdat)
166{ 193{
@@ -203,7 +230,7 @@ void register_page_bootmem_info_node(struct pglist_data *pgdat)
203 register_page_bootmem_info_section(pfn); 230 register_page_bootmem_info_section(pfn);
204 } 231 }
205} 232}
206#endif /* !CONFIG_SPARSEMEM_VMEMMAP */ 233#endif /* CONFIG_HAVE_BOOTMEM_INFO_NODE */
207 234
208static void grow_zone_span(struct zone *zone, unsigned long start_pfn, 235static void grow_zone_span(struct zone *zone, unsigned long start_pfn,
209 unsigned long end_pfn) 236 unsigned long end_pfn)