Re: BUG: Bad page state in process dirtyc0w_child
From: Gerald Schaefer
Date: Wed Sep 23 2020 - 09:39:51 EST
On Tue, 22 Sep 2020 19:03:50 +0200
Gerald Schaefer <gerald.schaefer@xxxxxxxxxxxxx> wrote:
> On Wed, 16 Sep 2020 16:28:06 +0200
> Heiko Carstens <hca@xxxxxxxxxxxxx> wrote:
>
> > On Sat, Sep 12, 2020 at 09:54:12PM -0400, Qian Cai wrote:
> > > Occasionally, running this LTP test will trigger an error below on
> > > s390:
> > > https://github.com/linux-test-project/ltp/blob/master/testcases/kernel/security/dirtyc0w/dirtyc0w.c
> > > https://github.com/linux-test-project/ltp/blob/master/testcases/kernel/security/dirtyc0w/dirtyc0w_child.c
> > >
> > > this .config:
> > > https://gitlab.com/cailca/linux-mm/-/blob/master/s390.config
> > >
> > > [ 6970.253173] LTP: starting dirtyc0w
> > > [ 6971.599102] BUG: Bad page state in process dirtyc0w_child pfn:8865d
> > > [ 6971.599867] page:000000001a8328d7 refcount:0 mapcount:0 mapping:0000000000000000 index:0x0 pfn:0x8865d
> > > [ 6971.599876] flags: 0x400000000008000e(referenced|uptodate|dirty|swapbacked)
> > > [ 6971.599886] raw: 400000000008000e 0000000000000100 0000000000000122 0000000000000000
> > > [ 6971.599893] raw: 0000000000000000 0000000000000000 ffffffff00000000 0000000000000000
> > > [ 6971.599900] page dumped because: PAGE_FLAGS_CHECK_AT_FREE flag(s) set
> > > [ 6971.599906] Modules linked in: loop kvm ip_tables x_tables dm_mirror dm_region_hash dm_log dm_mod [last unloaded: dummy_del_mod]
> > > [ 6971.599952] CPU: 1 PID: 65238 Comm: dirtyc0w_child Tainted: G O 5.9.0-rc4-next-20200909 #1
> > > [ 6971.599959] Hardware name: IBM 2964 N96 400 (z/VM 6.4.0)
> > > [ 6971.599964] Call Trace:
> > > [ 6971.599979] [<0000000073aec038>] show_stack+0x158/0x1f0
> > > [ 6971.599986] [<0000000073af724a>] dump_stack+0x1f2/0x238
> > > [ 6971.599994] [<0000000072ed086a>] bad_page+0x1ba/0x1c0
> > > [ 6971.600000] [<0000000072ed20c4>] free_pcp_prepare+0x4fc/0x658
> > > [ 6971.600006] [<0000000072ed96a6>] free_unref_page+0xae/0x158
> > > [ 6971.600013] [<0000000072e8286a>] unmap_page_range+0xb62/0x1df8
> > > [ 6971.600019] [<0000000072e83bbc>] unmap_single_vma+0xbc/0x1c8
> > > [ 6971.600025] [<0000000072e8418e>] zap_page_range+0x176/0x230
> > > [ 6971.600033] [<0000000072eece8e>] do_madvise+0xfde/0x1270
> > > [ 6971.600039] [<0000000072eed50a>] __s390x_sys_madvise+0x72/0x98
> > > [ 6971.600047] [<0000000073b1cce4>] system_call+0xdc/0x278
> > > [ 6971.600053] 2 locks held by dirtyc0w_child/65238:
> > > [ 6971.600058] #0: 000000013442fa18 (&mm->mmap_lock){++++}-{3:3}, at: do_madvise+0x17a/0x1270
> > > [ 6971.600432] #1: 00000001343f9060 (ptlock_ptr(page)#2){+.+.}-{2:2}, at: unmap_page_range+0x640/0x1df8
> > > [ 6971.600487] Disabling lock debugging due to kernel taint
> > >
> > > Once it happens, running it again will trigger in on another PFN.
> > >
> > > [39717.085115] BUG: Bad page state in process dirtyc0w_child pfn:af065
> > >
> > > Any thoughts?
> >
> > Alexander, Gerald, could you take a look?
>
> Thanks for reporting. From the header of dirtyc0w.c it seems that this
> is testing some gup behavior. Given that we have an issue with gup_fast
> on s390, this could be related. I'll try to reproduce and do more
> analysis.
>
> A fix for our gup_fast issue is also in linux-next now, as of 2020-09-20,
> but it was not yet included in your kernel version 5.9.0-rc4-next-20200909.
> So if this is related to the gup_fast issue, it should not occur again
> with linux-next kernels after 2020-09-20.
OK, I can now reproduce this, and unfortunately also with the gup_fast
fix, so it is something different. Bisecting is a bit hard, as it will
not always show immediately, sometimes takes up to an hour.
Still, I think I found the culprit, merge commit b25d1dc9474e "Merge
branch 'simplify-do_wp_page'". Without those 4 patches, it works fine,
running over night.
Not sure why this only shows on s390, should not be architecture-specific,
but we do often see subtle races earlier than others due to hypervisor
impact.
The first commit 09854ba94c6a ("mm: do_wp_page() simplification") already
introduces this error. The dirtyc0w_child test seems to play with cow
and racing madvise(MADV_DONTNEED), but I have not yet fully understood
it and also not the changes from commit 09854ba94c6a. As Linus already
mentioned in the merge commit message, this is some bad timing for such
a change, so I don't want to delay this further with trying to understand
it better before reporting. Maybe Peter or Linus can spot some obvious
issue.
One thing that seems strange to me is that the page flags from the
bad page state output are (uptodate|swapbacked), see below, or
(referenced|uptodate|dirty|swapbacked) in the original report. But IIUC,
that should not qualify for the "PAGE_FLAGS_CHECK_AT_FREE flag(s) set"
reason. So it seems that the flags have changed between check_free_page()
and __dump_page(), which would be very odd. Or maybe some issue with
compound pages, because __dump_page() looks at head->flags.
[ 1863.237707] BUG: Bad page state in process dirtyc0w_child pfn:58527d
[ 1863.237721] page:000000008866956b refcount:0 mapcount:0 mapping:0000000000000000 index:0x0 pfn:0x58527d
[ 1863.237727] flags: 0x3ffff00000080004(uptodate|swapbacked)
[ 1863.237734] raw: 3ffff00000080004 0000000000000100 0000000000000122 0000000000000000
[ 1863.237738] raw: 0000000000000000 0000000000000000 ffffffff00000000 0000000000000000
[ 1863.237742] page dumped because: PAGE_FLAGS_CHECK_AT_FREE flag(s) set
[ 1863.237745] Modules linked in:
[ 1863.237752] CPU: 16 PID: 9074 Comm: dirtyc0w_child Tainted: G B 5.9.0-rc6-00020-geff48ddeab78-dirty #104
[ 1863.237756] Hardware name: IBM 3906 M03 703 (LPAR)
[ 1863.237759] Call Trace:
[ 1863.237768] [<0000000000115f28>] show_stack+0x100/0x158
[ 1863.237775] [<000000000096b41a>] dump_stack+0xa2/0xd8
[ 1863.237781] [<00000000003d497c>] bad_page+0xdc/0x140
[ 1863.237785] [<00000000003d5b62>] free_pcp_prepare+0x31a/0x360
[ 1863.237789] [<00000000003d906a>] free_unref_page+0x32/0xb8
[ 1863.237794] [<00000000003b05f4>] zap_p4d_range+0x64c/0xcf8
[ 1863.237797] [<00000000003b0e7a>] unmap_page_range+0x9a/0x110
[ 1863.237801] [<00000000003b0f84>] unmap_single_vma+0x94/0x100
[ 1863.237805] [<00000000003b14c2>] zap_page_range+0x14a/0x1f0
[ 1863.237809] [<00000000003e3a24>] do_madvise+0x75c/0x918
[ 1863.237812] [<00000000003e3c06>] __s390x_sys_madvise+0x26/0x38
[ 1863.237817] [<0000000000d280d4>] system_call+0xe0/0x2c0
[ 1863.237820] INFO: lockdep is turned off.