diff options
Diffstat (limited to 'mm/sparse.c')
-rw-r--r-- | mm/sparse.c | 15 |
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 | ||
250 | static unsigned long *__init sparse_early_usemap_alloc(unsigned long pnum) | 250 | static 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 | ||