mm: hwpoison: fix action_result() to print out dirty/clean
action_result() fails to print out "dirty" even if an error occurred on
a dirty pagecache, because when we check PageDirty in action_result() it
was cleared after page isolation even if it's dirty before error
handling. This can break some applications that monitor this message,
so should be fixed.
There are several callers of action_result() except page_action(), but
either of them are not for LRU pages but for free pages or kernel pages,
so we don't have to consider dirty or not for them.
Note that PG_dirty can be set outside page locks as described in commit
6746aff74d
("HWPOISON: shmem: call set_page_dirty() with locked
page"), so this patch does not completely closes the race window, but
just narrows it.
Signed-off-by: Naoya Horiguchi <n-horiguchi@ah.jp.nec.com>
Reviewed-by: Andi Kleen <ak@linux.intel.com>
Cc: Tony Luck <tony.luck@intel.com>
Cc: "Jun'ichi Nomura" <j-nomura@ce.jp.nec.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
5de55b265a
commit
ff604cf6d4
1 changed files with 13 additions and 13 deletions
|
@ -781,16 +781,16 @@ static struct page_state {
|
||||||
{ compound, compound, "huge", me_huge_page },
|
{ compound, compound, "huge", me_huge_page },
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
{ sc|dirty, sc|dirty, "swapcache", me_swapcache_dirty },
|
{ sc|dirty, sc|dirty, "dirty swapcache", me_swapcache_dirty },
|
||||||
{ sc|dirty, sc, "swapcache", me_swapcache_clean },
|
{ sc|dirty, sc, "clean swapcache", me_swapcache_clean },
|
||||||
|
|
||||||
{ unevict|dirty, unevict|dirty, "unevictable LRU", me_pagecache_dirty},
|
{ unevict|dirty, unevict|dirty, "dirty unevictable LRU", me_pagecache_dirty },
|
||||||
{ unevict, unevict, "unevictable LRU", me_pagecache_clean},
|
{ unevict, unevict, "clean unevictable LRU", me_pagecache_clean },
|
||||||
|
|
||||||
{ mlock|dirty, mlock|dirty, "mlocked LRU", me_pagecache_dirty },
|
{ mlock|dirty, mlock|dirty, "dirty mlocked LRU", me_pagecache_dirty },
|
||||||
{ mlock, mlock, "mlocked LRU", me_pagecache_clean },
|
{ mlock, mlock, "clean mlocked LRU", me_pagecache_clean },
|
||||||
|
|
||||||
{ lru|dirty, lru|dirty, "LRU", me_pagecache_dirty },
|
{ lru|dirty, lru|dirty, "dirty LRU", me_pagecache_dirty },
|
||||||
{ lru|dirty, lru, "clean LRU", me_pagecache_clean },
|
{ lru|dirty, lru, "clean LRU", me_pagecache_clean },
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -812,14 +812,14 @@ static struct page_state {
|
||||||
#undef slab
|
#undef slab
|
||||||
#undef reserved
|
#undef reserved
|
||||||
|
|
||||||
|
/*
|
||||||
|
* "Dirty/Clean" indication is not 100% accurate due to the possibility of
|
||||||
|
* setting PG_dirty outside page lock. See also comment above set_page_dirty().
|
||||||
|
*/
|
||||||
static void action_result(unsigned long pfn, char *msg, int result)
|
static void action_result(unsigned long pfn, char *msg, int result)
|
||||||
{
|
{
|
||||||
struct page *page = pfn_to_page(pfn);
|
pr_err("MCE %#lx: %s page recovery: %s\n",
|
||||||
|
pfn, msg, action_name[result]);
|
||||||
printk(KERN_ERR "MCE %#lx: %s%s page recovery: %s\n",
|
|
||||||
pfn,
|
|
||||||
PageDirty(page) ? "dirty " : "",
|
|
||||||
msg, action_name[result]);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int page_action(struct page_state *ps, struct page *p,
|
static int page_action(struct page_state *ps, struct page *p,
|
||||||
|
|
Loading…
Reference in a new issue