mirror of
https://github.com/followmsi/android_kernel_google_msm.git
synced 2024-11-06 23:17:41 +00:00
mm: sparse: fix usemap allocation above node descriptor section
commit99ab7b1944
upstream. After commitf5bf18fa22
("bootmem/sparsemem: remove limit constraint in alloc_bootmem_section"), usemap allocations may easily be placed outside the optimal section that holds the node descriptor, even if there is space available in that section. This results in unnecessary hotplug dependencies that need to have the node unplugged before the section holding the usemap. The reason is that the bootmem allocator doesn't guarantee a linear search starting from the passed allocation goal but may start out at a much higher address absent an upper limit. Fix this by trying the allocation with the limit at the section end, then retry without if that fails. This keeps the fix fromf5bf18fa22
of not panicking if the allocation does not fit in the section, but still makes sure to try to stay within the section at first. [rewritten massively by Johannes to apply to 3.4] Change-Id: Ie40e51558b3abcaf0c0e72846928b84fe17f055a Signed-off-by: Yinghai Lu <yinghai@kernel.org> Signed-off-by: Johannes Weiner <hannes@cmpxchg.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
84a5789882
commit
c7c94a6913
1 changed files with 6 additions and 2 deletions
|
@ -766,13 +766,17 @@ void * __init alloc_bootmem_section(unsigned long size,
|
||||||
unsigned long section_nr)
|
unsigned long section_nr)
|
||||||
{
|
{
|
||||||
bootmem_data_t *bdata;
|
bootmem_data_t *bdata;
|
||||||
unsigned long pfn, goal;
|
unsigned long pfn, goal, limit;
|
||||||
|
|
||||||
pfn = section_nr_to_pfn(section_nr);
|
pfn = section_nr_to_pfn(section_nr);
|
||||||
goal = pfn << PAGE_SHIFT;
|
goal = pfn << PAGE_SHIFT;
|
||||||
|
limit = section_nr_to_pfn(section_nr + 1) << PAGE_SHIFT;
|
||||||
bdata = &bootmem_node_data[early_pfn_to_nid(pfn)];
|
bdata = &bootmem_node_data[early_pfn_to_nid(pfn)];
|
||||||
|
|
||||||
return alloc_bootmem_core(bdata, size, SMP_CACHE_BYTES, goal, 0);
|
if (goal + size > limit)
|
||||||
|
limit = 0;
|
||||||
|
|
||||||
|
return alloc_bootmem_core(bdata, size, SMP_CACHE_BYTES, goal, limit);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue