aboutsummaryrefslogtreecommitdiffstats
path: root/mm/page_alloc.c
diff options
context:
space:
mode:
Diffstat (limited to 'mm/page_alloc.c')
-rw-r--r--mm/page_alloc.c30
1 files changed, 30 insertions, 0 deletions
diff --git a/mm/page_alloc.c b/mm/page_alloc.c
index c677c1506d73..e75865d58ba7 100644
--- a/mm/page_alloc.c
+++ b/mm/page_alloc.c
@@ -8096,3 +8096,33 @@ bool is_free_buddy_page(struct page *page)
8096 8096
8097 return order < MAX_ORDER; 8097 return order < MAX_ORDER;
8098} 8098}
8099
8100#ifdef CONFIG_MEMORY_FAILURE
8101/*
8102 * Set PG_hwpoison flag if a given page is confirmed to be a free page. This
8103 * test is performed under the zone lock to prevent a race against page
8104 * allocation.
8105 */
8106bool set_hwpoison_free_buddy_page(struct page *page)
8107{
8108 struct zone *zone = page_zone(page);
8109 unsigned long pfn = page_to_pfn(page);
8110 unsigned long flags;
8111 unsigned int order;
8112 bool hwpoisoned = false;
8113
8114 spin_lock_irqsave(&zone->lock, flags);
8115 for (order = 0; order < MAX_ORDER; order++) {
8116 struct page *page_head = page - (pfn & ((1 << order) - 1));
8117
8118 if (PageBuddy(page_head) && page_order(page_head) >= order) {
8119 if (!TestSetPageHWPoison(page))
8120 hwpoisoned = true;
8121 break;
8122 }
8123 }
8124 spin_unlock_irqrestore(&zone->lock, flags);
8125
8126 return hwpoisoned;
8127}
8128#endif