aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--mm/sparse.c15
1 files changed, 13 insertions, 2 deletions
diff --git a/mm/sparse.c b/mm/sparse.c
index 5398d48c360a..08f053218ee8 100644
--- a/mm/sparse.c
+++ b/mm/sparse.c
@@ -249,11 +249,22 @@ static unsigned long *__kmalloc_section_usemap(void)
249 249
250static unsigned long *__init sparse_early_usemap_alloc(unsigned long pnum) 250static unsigned long *__init sparse_early_usemap_alloc(unsigned long pnum)
251{ 251{
252 unsigned long *usemap; 252 unsigned long *usemap, section_nr;
253 struct mem_section *ms = __nr_to_section(pnum); 253 struct mem_section *ms = __nr_to_section(pnum);
254 int nid = sparse_early_nid(ms); 254 int nid = sparse_early_nid(ms);
255 struct pglist_data *pgdat = NODE_DATA(nid);
255 256
256 usemap = alloc_bootmem_node(NODE_DATA(nid), usemap_size()); 257 /*
258 * Usemap's page can't be freed until freeing other sections
259 * which use it. And, Pgdat has same feature.
260 * If section A has pgdat and section B has usemap for other
261 * sections (includes section A), both sections can't be removed,
262 * because there is the dependency each other.
263 * To solve above issue, this collects all usemap on the same section
264 * which has pgdat.
265 */
266 section_nr = pfn_to_section_nr(__pa(pgdat) >> PAGE_SHIFT);
267 usemap = alloc_bootmem_section(usemap_size(), section_nr);
257 if (usemap) 268 if (usemap)
258 return usemap; 269 return usemap;
259 270