aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/linux/mm.h1
-rw-r--r--mm/memory-failure.c22
2 files changed, 23 insertions, 0 deletions
diff --git a/include/linux/mm.h b/include/linux/mm.h
index 9d65ae4ba0e0..68c84bb2ad3f 100644
--- a/include/linux/mm.h
+++ b/include/linux/mm.h
@@ -1335,6 +1335,7 @@ extern void memory_failure(unsigned long pfn, int trapno);
1335extern int __memory_failure(unsigned long pfn, int trapno, int ref); 1335extern int __memory_failure(unsigned long pfn, int trapno, int ref);
1336extern int sysctl_memory_failure_early_kill; 1336extern int sysctl_memory_failure_early_kill;
1337extern int sysctl_memory_failure_recovery; 1337extern int sysctl_memory_failure_recovery;
1338extern void shake_page(struct page *p);
1338extern atomic_long_t mce_bad_pages; 1339extern atomic_long_t mce_bad_pages;
1339 1340
1340#endif /* __KERNEL__ */ 1341#endif /* __KERNEL__ */
diff --git a/mm/memory-failure.c b/mm/memory-failure.c
index 50d4f8d7024a..38fcbb22eab9 100644
--- a/mm/memory-failure.c
+++ b/mm/memory-failure.c
@@ -83,6 +83,28 @@ static int kill_proc_ao(struct task_struct *t, unsigned long addr, int trapno,
83} 83}
84 84
85/* 85/*
86 * When a unknown page type is encountered drain as many buffers as possible
87 * in the hope to turn the page into a LRU or free page, which we can handle.
88 */
89void shake_page(struct page *p)
90{
91 if (!PageSlab(p)) {
92 lru_add_drain_all();
93 if (PageLRU(p))
94 return;
95 drain_all_pages();
96 if (PageLRU(p) || is_free_buddy_page(p))
97 return;
98 }
99 /*
100 * Could call shrink_slab here (which would also
101 * shrink other caches). Unfortunately that might
102 * also access the corrupted page, which could be fatal.
103 */
104}
105EXPORT_SYMBOL_GPL(shake_page);
106
107/*
86 * Kill all processes that have a poisoned page mapped and then isolate 108 * Kill all processes that have a poisoned page mapped and then isolate
87 * the page. 109 * the page.
88 * 110 *