forked from Minki/linux
thp: fix unsuitable behavior for hwpoisoned tail page
When a tail page of THP is poisoned, memory-failure will do nothing except setting PG_hwpoison, while the expected behavior is that the process, who is using the poisoned tail page, should be killed. The above problem is caused by lru check of the poisoned tail page of THP. Because PG_lru flag is only set on the head page of THP, the check always consider the poisoned tail page as NON lru page. So the lru check for the tail page of THP should be avoided, as like as hugetlb. This patch adds !PageTransCompound() before lru check for THP, because of the check (!PageHuge() && !PageTransCompound()) the whole branch could be optimized away at build time when both hugetlbfs and THP are set with "N" (or in archs not supporting either of those). [akpm@linux-foundation.org: fix unrelated typo in shake_page() comment] Signed-off-by: Jin Dongming <jin.dongming@np.css.fujitsu.com> Reviewed-by: Hidetoshi Seto <seto.hidetoshi@jp.fujitsu.com> Cc: Andrea Arcangeli <aarcange@redhat.com> Cc: Andi Kleen <andi@firstfloor.org> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
a6d30dddae
commit
af241a0834
@ -233,8 +233,8 @@ void shake_page(struct page *p, int access)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Only all shrink_slab here (which would also
|
* Only call shrink_slab here (which would also shrink other caches) if
|
||||||
* shrink other caches) if access is not potentially fatal.
|
* access is not potentially fatal.
|
||||||
*/
|
*/
|
||||||
if (access) {
|
if (access) {
|
||||||
int nr;
|
int nr;
|
||||||
@ -1065,19 +1065,22 @@ int __memory_failure(unsigned long pfn, int trapno, int flags)
|
|||||||
* The check (unnecessarily) ignores LRU pages being isolated and
|
* The check (unnecessarily) ignores LRU pages being isolated and
|
||||||
* walked by the page reclaim code, however that's not a big loss.
|
* walked by the page reclaim code, however that's not a big loss.
|
||||||
*/
|
*/
|
||||||
if (!PageLRU(p) && !PageHuge(p))
|
if (!PageHuge(p) && !PageTransCompound(p)) {
|
||||||
shake_page(p, 0);
|
if (!PageLRU(p))
|
||||||
if (!PageLRU(p) && !PageHuge(p)) {
|
shake_page(p, 0);
|
||||||
/*
|
if (!PageLRU(p)) {
|
||||||
* shake_page could have turned it free.
|
/*
|
||||||
*/
|
* shake_page could have turned it free.
|
||||||
if (is_free_buddy_page(p)) {
|
*/
|
||||||
action_result(pfn, "free buddy, 2nd try", DELAYED);
|
if (is_free_buddy_page(p)) {
|
||||||
return 0;
|
action_result(pfn, "free buddy, 2nd try",
|
||||||
|
DELAYED);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
action_result(pfn, "non LRU", IGNORED);
|
||||||
|
put_page(p);
|
||||||
|
return -EBUSY;
|
||||||
}
|
}
|
||||||
action_result(pfn, "non LRU", IGNORED);
|
|
||||||
put_page(p);
|
|
||||||
return -EBUSY;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
Loading…
Reference in New Issue
Block a user