Am 28.11.2017 um 12:42 hat Kevin Wolf geschrieben: > Am 28.11.2017 um 06:43 hat Jeff Cody geschrieben: > > On Tue, Nov 28, 2017 at 12:29:09AM +0100, Kevin Wolf wrote: > > > Am 23.11.2017 um 18:57 hat Fam Zheng geschrieben: > > > > Jeff's block job patch made the latent drain bug visible, and I find > > > > this > > > > patch, which by itself also makes some sense, can hide it again. :) > > > > With it > > > > applied we are at least back to the ground where patchew's iotests (make > > > > docker-test-block@fedora) can pass. > > > > > > > > The real bug is that in the middle of bdrv_parent_drained_end(), bs's > > > > parent > > > > list changes. One drained_end call before the mirror_exit() already did > > > > one > > > > blk_root_drained_end(), a second drained_end on an updated parent node > > > > can do > > > > another same blk_root_drained_end(), making it unbalanced with > > > > blk_root_drained_begin(). This is shown by the following three > > > > backtraces as > > > > captured by rr with a crashed "qemu-img commit", essentially the same > > > > as in > > > > the failed iotest 020: > > > > > > My conclusion what really happens in 020 is that we have a graph like > > > this: > > > > > > mirror target BB --+ > > > | > > > v > > > qemu-img BB -> mirror_top_bs -> overlay -> base > > > > > > bdrv_drained_end(base) results in it being available for requests again, > > > so it calls bdrv_parent_drained_end() for overlay. While draining > > > itself, the mirror job completes and changes the BdrvChild between > > > mirror_top_bs and overlay (which is currently being drained) to point to > > > base instead. After returning, QLIST_FOREACH() continues to iterate the > > > parents of base instead of those of overlay, resulting in a second > > > blk_drained_end() for the mirror target BB. > > > > > > This instance can be fixed relatively easily (see below) by using > > > QLIST_FOREACH_SAFE() instead. > > > > > > However, I'm not sure if all problems with the graph change can be > > > solved this way and whether we can really allow graph changes while > > > iterating the graph for bdrv_drained_begin/end. Not allowing it would > > > require some more serious changes to the block jobs that delays their > > > completion until after bdrv_drain_end() has finished (not sure how to > > > even get a callback at that point...) > > > > > > > That is at least part of what is causing the segfaults that I am still > > seeing (after your patch): > > > > We enter bdrv_drain_recurse(), and the BDS has been reaped: > > Not sure which test case this is referring to, probably 097 as that's > the next one in your list? > > Anyway, test cases 097 and 176 can be fixed for me by keeping some > additional references. This quick fix is probably not quite correct > according to the comment in bdrv_drain_recurse() because bdrv_ref/unref > are only allowed in the main loop thread. > > Also, case 141 is still failing.
As for 141, this one just hangs now because the test case sets speed=1 and so the job throttling decides to sleep for a few hours. We used to interrupt block_job_sleep_ns(), now we don't any more. I think we need to allow this again. Kevin