diff options
Diffstat (limited to 'mm')
-rw-r--r-- | mm/memory-failure.c | 38 |
1 files changed, 38 insertions, 0 deletions
diff --git a/mm/memory-failure.c b/mm/memory-failure.c index 1ec68c80788e..fee648b9d393 100644 --- a/mm/memory-failure.c +++ b/mm/memory-failure.c | |||
@@ -920,6 +920,22 @@ static int hwpoison_user_mappings(struct page *p, unsigned long pfn, | |||
920 | return ret; | 920 | return ret; |
921 | } | 921 | } |
922 | 922 | ||
923 | static void set_page_hwpoison_huge_page(struct page *hpage) | ||
924 | { | ||
925 | int i; | ||
926 | int nr_pages = 1 << compound_order(hpage); | ||
927 | for (i = 0; i < nr_pages; i++) | ||
928 | SetPageHWPoison(hpage + i); | ||
929 | } | ||
930 | |||
931 | static void clear_page_hwpoison_huge_page(struct page *hpage) | ||
932 | { | ||
933 | int i; | ||
934 | int nr_pages = 1 << compound_order(hpage); | ||
935 | for (i = 0; i < nr_pages; i++) | ||
936 | ClearPageHWPoison(hpage + i); | ||
937 | } | ||
938 | |||
923 | int __memory_failure(unsigned long pfn, int trapno, int flags) | 939 | int __memory_failure(unsigned long pfn, int trapno, int flags) |
924 | { | 940 | { |
925 | struct page_state *ps; | 941 | struct page_state *ps; |
@@ -1014,6 +1030,26 @@ int __memory_failure(unsigned long pfn, int trapno, int flags) | |||
1014 | return 0; | 1030 | return 0; |
1015 | } | 1031 | } |
1016 | 1032 | ||
1033 | /* | ||
1034 | * For error on the tail page, we should set PG_hwpoison | ||
1035 | * on the head page to show that the hugepage is hwpoisoned | ||
1036 | */ | ||
1037 | if (PageTail(p) && TestSetPageHWPoison(hpage)) { | ||
1038 | action_result(pfn, "hugepage already hardware poisoned", | ||
1039 | IGNORED); | ||
1040 | unlock_page(hpage); | ||
1041 | put_page(hpage); | ||
1042 | return 0; | ||
1043 | } | ||
1044 | /* | ||
1045 | * Set PG_hwpoison on all pages in an error hugepage, | ||
1046 | * because containment is done in hugepage unit for now. | ||
1047 | * Since we have done TestSetPageHWPoison() for the head page with | ||
1048 | * page lock held, we can safely set PG_hwpoison bits on tail pages. | ||
1049 | */ | ||
1050 | if (PageHuge(p)) | ||
1051 | set_page_hwpoison_huge_page(hpage); | ||
1052 | |||
1017 | wait_on_page_writeback(p); | 1053 | wait_on_page_writeback(p); |
1018 | 1054 | ||
1019 | /* | 1055 | /* |
@@ -1118,6 +1154,8 @@ int unpoison_memory(unsigned long pfn) | |||
1118 | atomic_long_dec(&mce_bad_pages); | 1154 | atomic_long_dec(&mce_bad_pages); |
1119 | freeit = 1; | 1155 | freeit = 1; |
1120 | } | 1156 | } |
1157 | if (PageHuge(p)) | ||
1158 | clear_page_hwpoison_huge_page(page); | ||
1121 | unlock_page(page); | 1159 | unlock_page(page); |
1122 | 1160 | ||
1123 | put_page(page); | 1161 | put_page(page); |