diff options
| author | Stefan Hajnoczi | 2017-11-10 16:19:34 +0100 |
|---|---|---|
| committer | Stefan Hajnoczi | 2017-11-13 15:02:09 +0100 |
| commit | dc868fb03b9b829ed9d2ecdae0fcc12f3fe19b4f (patch) | |
| tree | 21130555d5c54a977907bd8e7e8e1fc836fc6874 /block/throttle-groups.c | |
| parent | block: all I/O should be completed before removing throttle timers. (diff) | |
| download | qemu-dc868fb03b9b829ed9d2ecdae0fcc12f3fe19b4f.tar.gz qemu-dc868fb03b9b829ed9d2ecdae0fcc12f3fe19b4f.tar.xz qemu-dc868fb03b9b829ed9d2ecdae0fcc12f3fe19b4f.zip | |
throttle-groups: drain before detaching ThrottleState
I/O requests hang after stop/cont commands at least since QEMU 2.10.0
with -drive iops=100:
(guest)$ dd if=/dev/zero of=/dev/vdb oflag=direct count=1000
(qemu) stop
(qemu) cont
...I/O is stuck...
This happens because blk_set_aio_context() detaches the ThrottleState
while requests may still be in flight:
if (tgm->throttle_state) {
throttle_group_detach_aio_context(tgm);
throttle_group_attach_aio_context(tgm, new_context);
}
This patch encloses the detach/attach calls in a drained region so no
I/O request is left hanging. Also add assertions so we don't make the
same mistake again in the future.
Reported-by: Yongxue Hong <yhong@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Reviewed-by: Alberto Garcia <berto@igalia.com>
Message-id: 20171110151934.16883-1-stefanha@redhat.com
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Diffstat (limited to 'block/throttle-groups.c')
| -rw-r--r-- | block/throttle-groups.c | 6 |
1 files changed, 6 insertions, 0 deletions
diff --git a/block/throttle-groups.c b/block/throttle-groups.c index b291a88481..2587f19ca3 100644 --- a/block/throttle-groups.c +++ b/block/throttle-groups.c @@ -594,6 +594,12 @@ void throttle_group_attach_aio_context(ThrottleGroupMember *tgm, void throttle_group_detach_aio_context(ThrottleGroupMember *tgm) { ThrottleTimers *tt = &tgm->throttle_timers; + + /* Requests must have been drained */ + assert(tgm->pending_reqs[0] == 0 && tgm->pending_reqs[1] == 0); + assert(qemu_co_queue_empty(&tgm->throttled_reqs[0])); + assert(qemu_co_queue_empty(&tgm->throttled_reqs[1])); + throttle_timers_detach_aio_context(tt); tgm->aio_context = NULL; } |
