diff options
author | Wu Fengguang <fengguang.wu@intel.com> | 2009-12-16 06:19:58 -0500 |
---|---|---|
committer | Andi Kleen <ak@linux.intel.com> | 2009-12-16 06:19:58 -0500 |
commit | 8d22ba1b74aa9420b6032d856446564fb21f8090 (patch) | |
tree | 6c2e2d27e81d784faa0481500f1cecc613ff1167 | |
parent | 95d01fc664b9476e0d18e3d745bb209a42a33588 (diff) |
HWPOISON: detect free buddy pages explicitly
Most free pages in the buddy system have no PG_buddy set.
Introduce is_free_buddy_page() for detecting them reliably.
CC: Nick Piggin <npiggin@suse.de>
CC: Mel Gorman <mel@linux.vnet.ibm.com>
Signed-off-by: Wu Fengguang <fengguang.wu@intel.com>
Signed-off-by: Andi Kleen <ak@linux.intel.com>
-rw-r--r-- | mm/internal.h | 3 | ||||
-rw-r--r-- | mm/memory-failure.c | 9 | ||||
-rw-r--r-- | mm/page_alloc.c | 21 |
3 files changed, 31 insertions, 2 deletions
diff --git a/mm/internal.h b/mm/internal.h index 4fe67a162cb4..49b2ff776b78 100644 --- a/mm/internal.h +++ b/mm/internal.h | |||
@@ -50,6 +50,9 @@ extern void putback_lru_page(struct page *page); | |||
50 | */ | 50 | */ |
51 | extern void __free_pages_bootmem(struct page *page, unsigned int order); | 51 | extern void __free_pages_bootmem(struct page *page, unsigned int order); |
52 | extern void prep_compound_page(struct page *page, unsigned long order); | 52 | extern void prep_compound_page(struct page *page, unsigned long order); |
53 | #ifdef CONFIG_MEMORY_FAILURE | ||
54 | extern bool is_free_buddy_page(struct page *page); | ||
55 | #endif | ||
53 | 56 | ||
54 | 57 | ||
55 | /* | 58 | /* |
diff --git a/mm/memory-failure.c b/mm/memory-failure.c index 676ab394200e..5055b940df5f 100644 --- a/mm/memory-failure.c +++ b/mm/memory-failure.c | |||
@@ -807,8 +807,13 @@ int __memory_failure(unsigned long pfn, int trapno, int flags) | |||
807 | */ | 807 | */ |
808 | if (!(flags & MF_COUNT_INCREASED) && | 808 | if (!(flags & MF_COUNT_INCREASED) && |
809 | !get_page_unless_zero(compound_head(p))) { | 809 | !get_page_unless_zero(compound_head(p))) { |
810 | action_result(pfn, "free or high order kernel", IGNORED); | 810 | if (is_free_buddy_page(p)) { |
811 | return PageBuddy(compound_head(p)) ? 0 : -EBUSY; | 811 | action_result(pfn, "free buddy", DELAYED); |
812 | return 0; | ||
813 | } else { | ||
814 | action_result(pfn, "high order kernel", IGNORED); | ||
815 | return -EBUSY; | ||
816 | } | ||
812 | } | 817 | } |
813 | 818 | ||
814 | /* | 819 | /* |
diff --git a/mm/page_alloc.c b/mm/page_alloc.c index 59d2e88fb47c..6867b4d391fd 100644 --- a/mm/page_alloc.c +++ b/mm/page_alloc.c | |||
@@ -5081,3 +5081,24 @@ __offline_isolated_pages(unsigned long start_pfn, unsigned long end_pfn) | |||
5081 | spin_unlock_irqrestore(&zone->lock, flags); | 5081 | spin_unlock_irqrestore(&zone->lock, flags); |
5082 | } | 5082 | } |
5083 | #endif | 5083 | #endif |
5084 | |||
5085 | #ifdef CONFIG_MEMORY_FAILURE | ||
5086 | bool is_free_buddy_page(struct page *page) | ||
5087 | { | ||
5088 | struct zone *zone = page_zone(page); | ||
5089 | unsigned long pfn = page_to_pfn(page); | ||
5090 | unsigned long flags; | ||
5091 | int order; | ||
5092 | |||
5093 | spin_lock_irqsave(&zone->lock, flags); | ||
5094 | for (order = 0; order < MAX_ORDER; order++) { | ||
5095 | struct page *page_head = page - (pfn & ((1 << order) - 1)); | ||
5096 | |||
5097 | if (PageBuddy(page_head) && page_order(page_head) >= order) | ||
5098 | break; | ||
5099 | } | ||
5100 | spin_unlock_irqrestore(&zone->lock, flags); | ||
5101 | |||
5102 | return order < MAX_ORDER; | ||
5103 | } | ||
5104 | #endif | ||