diff options
author | Minchan Kim <minchan@kernel.org> | 2012-10-08 19:32:14 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2012-10-09 03:22:46 -0400 |
commit | 41d575ad4a511b71a4a41c8313004212f5c229b1 (patch) | |
tree | 04d9fe90086a7c4be7b680c3adc592086baccfe5 /mm/page_isolation.c | |
parent | 95e3441248053fc06bbb1dbbd34409a84211619e (diff) |
memory-hotplug: bug fix race between isolation and allocation
Like below, memory-hotplug makes race between page-isolation
and page-allocation so it can hit BUG_ON in __offline_isolated_pages.
CPU A CPU B
start_isolate_page_range
set_migratetype_isolate
spin_lock_irqsave(zone->lock)
free_hot_cold_page(Page A)
/* without zone->lock */
migratetype = get_pageblock_migratetype(Page A);
/*
* Page could be moved into MIGRATE_MOVABLE
* of per_cpu_pages
*/
list_add_tail(&page->lru, &pcp->lists[migratetype]);
set_pageblock_isolate
move_freepages_block
drain_all_pages
/* Page A could be in MIGRATE_MOVABLE of free_list. */
check_pages_isolated
__test_page_isolated_in_pageblock
/*
* We can't catch freed page which
* is free_list[MIGRATE_MOVABLE]
*/
if (PageBuddy(page A))
pfn += 1 << page_order(page A);
/* So, Page A could be allocated */
__offline_isolated_pages
/*
* BUG_ON hit or offline page
* which is used by someone
*/
BUG_ON(!PageBuddy(page A));
This patch checks page's migratetype in freelist in
__test_page_isolated_in_pageblock. So now
__test_page_isolated_in_pageblock can check the page caused by above race
and can fail of memory offlining.
Signed-off-by: Minchan Kim <minchan@kernel.org>
Acked-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Reviewed-by: Yasuaki Ishimatsu <isimatu.yasuaki@jp.fujitsu.com>
Acked-by: Mel Gorman <mgorman@suse.de>
Cc: Xishi Qiu <qiuxishi@huawei.com>
Cc: Wen Congyang <wency@cn.fujitsu.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm/page_isolation.c')
-rw-r--r-- | mm/page_isolation.c | 5 |
1 files changed, 4 insertions, 1 deletions
diff --git a/mm/page_isolation.c b/mm/page_isolation.c index 9c03dca8c2ad..6744235d2d0e 100644 --- a/mm/page_isolation.c +++ b/mm/page_isolation.c | |||
@@ -200,8 +200,11 @@ __test_page_isolated_in_pageblock(unsigned long pfn, unsigned long end_pfn) | |||
200 | continue; | 200 | continue; |
201 | } | 201 | } |
202 | page = pfn_to_page(pfn); | 202 | page = pfn_to_page(pfn); |
203 | if (PageBuddy(page)) | 203 | if (PageBuddy(page)) { |
204 | if (get_freepage_migratetype(page) != MIGRATE_ISOLATE) | ||
205 | break; | ||
204 | pfn += 1 << page_order(page); | 206 | pfn += 1 << page_order(page); |
207 | } | ||
205 | else if (page_count(page) == 0 && | 208 | else if (page_count(page) == 0 && |
206 | get_freepage_migratetype(page) == MIGRATE_ISOLATE) | 209 | get_freepage_migratetype(page) == MIGRATE_ISOLATE) |
207 | pfn += 1; | 210 | pfn += 1; |