diff options
Diffstat (limited to 'mm/memory-failure.c')
| -rw-r--r-- | mm/memory-failure.c | 15 |
1 files changed, 10 insertions, 5 deletions
diff --git a/mm/memory-failure.c b/mm/memory-failure.c index fee648b9d393..473f15a3356d 100644 --- a/mm/memory-failure.c +++ b/mm/memory-failure.c | |||
| @@ -942,6 +942,7 @@ int __memory_failure(unsigned long pfn, int trapno, int flags) | |||
| 942 | struct page *p; | 942 | struct page *p; |
| 943 | struct page *hpage; | 943 | struct page *hpage; |
| 944 | int res; | 944 | int res; |
| 945 | unsigned int nr_pages; | ||
| 945 | 946 | ||
| 946 | if (!sysctl_memory_failure_recovery) | 947 | if (!sysctl_memory_failure_recovery) |
| 947 | panic("Memory failure from trap %d on page %lx", trapno, pfn); | 948 | panic("Memory failure from trap %d on page %lx", trapno, pfn); |
| @@ -960,7 +961,8 @@ int __memory_failure(unsigned long pfn, int trapno, int flags) | |||
| 960 | return 0; | 961 | return 0; |
| 961 | } | 962 | } |
| 962 | 963 | ||
| 963 | atomic_long_add(1, &mce_bad_pages); | 964 | nr_pages = 1 << compound_order(hpage); |
| 965 | atomic_long_add(nr_pages, &mce_bad_pages); | ||
| 964 | 966 | ||
| 965 | /* | 967 | /* |
| 966 | * We need/can do nothing about count=0 pages. | 968 | * We need/can do nothing about count=0 pages. |
| @@ -1024,7 +1026,7 @@ int __memory_failure(unsigned long pfn, int trapno, int flags) | |||
| 1024 | } | 1026 | } |
| 1025 | if (hwpoison_filter(p)) { | 1027 | if (hwpoison_filter(p)) { |
| 1026 | if (TestClearPageHWPoison(p)) | 1028 | if (TestClearPageHWPoison(p)) |
| 1027 | atomic_long_dec(&mce_bad_pages); | 1029 | atomic_long_sub(nr_pages, &mce_bad_pages); |
| 1028 | unlock_page(hpage); | 1030 | unlock_page(hpage); |
| 1029 | put_page(hpage); | 1031 | put_page(hpage); |
| 1030 | return 0; | 1032 | return 0; |
| @@ -1123,6 +1125,7 @@ int unpoison_memory(unsigned long pfn) | |||
| 1123 | struct page *page; | 1125 | struct page *page; |
| 1124 | struct page *p; | 1126 | struct page *p; |
| 1125 | int freeit = 0; | 1127 | int freeit = 0; |
| 1128 | unsigned int nr_pages; | ||
| 1126 | 1129 | ||
| 1127 | if (!pfn_valid(pfn)) | 1130 | if (!pfn_valid(pfn)) |
| 1128 | return -ENXIO; | 1131 | return -ENXIO; |
| @@ -1135,9 +1138,11 @@ int unpoison_memory(unsigned long pfn) | |||
| 1135 | return 0; | 1138 | return 0; |
| 1136 | } | 1139 | } |
| 1137 | 1140 | ||
| 1141 | nr_pages = 1 << compound_order(page); | ||
| 1142 | |||
| 1138 | if (!get_page_unless_zero(page)) { | 1143 | if (!get_page_unless_zero(page)) { |
| 1139 | if (TestClearPageHWPoison(p)) | 1144 | if (TestClearPageHWPoison(p)) |
| 1140 | atomic_long_dec(&mce_bad_pages); | 1145 | atomic_long_sub(nr_pages, &mce_bad_pages); |
| 1141 | pr_debug("MCE: Software-unpoisoned free page %#lx\n", pfn); | 1146 | pr_debug("MCE: Software-unpoisoned free page %#lx\n", pfn); |
| 1142 | return 0; | 1147 | return 0; |
| 1143 | } | 1148 | } |
| @@ -1149,9 +1154,9 @@ int unpoison_memory(unsigned long pfn) | |||
| 1149 | * the PG_hwpoison page will be caught and isolated on the entrance to | 1154 | * the PG_hwpoison page will be caught and isolated on the entrance to |
| 1150 | * the free buddy page pool. | 1155 | * the free buddy page pool. |
| 1151 | */ | 1156 | */ |
| 1152 | if (TestClearPageHWPoison(p)) { | 1157 | if (TestClearPageHWPoison(page)) { |
| 1153 | pr_debug("MCE: Software-unpoisoned page %#lx\n", pfn); | 1158 | pr_debug("MCE: Software-unpoisoned page %#lx\n", pfn); |
| 1154 | atomic_long_dec(&mce_bad_pages); | 1159 | atomic_long_sub(nr_pages, &mce_bad_pages); |
| 1155 | freeit = 1; | 1160 | freeit = 1; |
| 1156 | } | 1161 | } |
| 1157 | if (PageHuge(p)) | 1162 | if (PageHuge(p)) |
