| #!/usr/bin/env bash |
| # group: rw |
| # |
| # Test exiting qemu while jobs are still running |
| # |
| # Copyright (C) 2017 Red Hat, Inc. |
| # |
| # This program is free software; you can redistribute it and/or modify |
| # it under the terms of the GNU General Public License as published by |
| # the Free Software Foundation; either version 2 of the License, or |
| # (at your option) any later version. |
| # |
| # This program is distributed in the hope that it will be useful, |
| # but WITHOUT ANY WARRANTY; without even the implied warranty of |
| # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| # GNU General Public License for more details. |
| # |
| # You should have received a copy of the GNU General Public License |
| # along with this program. If not, see <http://www.gnu.org/licenses/>. |
| # |
| |
| # creator |
| owner=kwolf@redhat.com |
| |
| seq=`basename $0` |
| echo "QA output created by $seq" |
| |
| status=1 # failure is the default! |
| |
| _cleanup() |
| { |
| _rm_test_img "${TEST_IMG}.mid" |
| _rm_test_img "${TEST_IMG}.copy" |
| _cleanup_test_img |
| _cleanup_qemu |
| |
| if [ -f "$TEST_DIR/qsd.pid" ]; then |
| kill -SIGKILL "$(cat "$TEST_DIR/qsd.pid")" |
| rm -f "$TEST_DIR/qsd.pid" |
| fi |
| rm -f "$SOCK_DIR/qsd.sock" |
| } |
| trap "_cleanup; exit \$status" 0 1 2 3 15 |
| |
| # get standard environment, filters and checks |
| . ./common.rc |
| . ./common.filter |
| . ./common.qemu |
| |
| _supported_fmt qcow2 |
| _supported_proto file |
| _supported_os Linux |
| |
| size=$((64 * 1048576)) |
| TEST_IMG="${TEST_IMG}.base" _make_test_img $size |
| |
| echo |
| echo === Starting VM === |
| echo |
| |
| qemu_comm_method="qmp" |
| |
| _launch_qemu \ |
| -drive file="${TEST_IMG}.base",cache=$CACHEMODE,aio=$AIOMODE,driver=$IMGFMT,id=disk |
| h=$QEMU_HANDLE |
| _send_qemu_cmd $h "{ 'execute': 'qmp_capabilities' }" 'return' |
| |
| echo |
| echo === Creating backing chain === |
| echo |
| |
| _send_qemu_cmd $h \ |
| "{ 'execute': 'blockdev-snapshot-sync', |
| 'arguments': { 'device': 'disk', |
| 'snapshot-file': '$TEST_IMG.mid', |
| 'format': '$IMGFMT', |
| 'mode': 'absolute-paths' } }" \ |
| "return" |
| |
| _send_qemu_cmd $h \ |
| "{ 'execute': 'human-monitor-command', |
| 'arguments': { 'command-line': |
| 'qemu-io disk \"write 0 4M\"' } }" \ |
| "return" |
| |
| _send_qemu_cmd $h \ |
| "{ 'execute': 'blockdev-snapshot-sync', |
| 'arguments': { 'device': 'disk', |
| 'snapshot-file': '$TEST_IMG', |
| 'format': '$IMGFMT', |
| 'mode': 'absolute-paths' } }" \ |
| "return" |
| |
| echo |
| echo === Start commit job and exit qemu === |
| echo |
| |
| # Note that the reference output intentionally includes the 'offset' field in |
| # BLOCK_JOB_* events for all of the following block jobs. They are predictable |
| # and any change in the offsets would hint at a bug in the job throttling code. |
| # |
| # In order to achieve these predictable offsets, all of the following tests |
| # use speed=65536. Each job will perform exactly one iteration before it has |
| # to sleep at least for a second, which is plenty of time for the 'quit' QMP |
| # command to be received (after receiving the command, the rest runs |
| # synchronously, so jobs can arbitrarily continue or complete). |
| # |
| # The buffer size for commit and streaming is 512k (waiting for 8 seconds after |
| # the first request), for active commit and mirror it's large enough to cover |
| # the full 4M, and for backup it's the qcow2 cluster size, which we know is |
| # 64k. As all of these are at least as large as the speed, we are sure that the |
| # offset advances exactly once before qemu exits. |
| |
| _send_qemu_cmd $h \ |
| "{ 'execute': 'block-commit', |
| 'arguments': { 'device': 'disk', |
| 'base':'$TEST_IMG.base', |
| 'top': '$TEST_IMG.mid', |
| 'speed': 65536 } }" \ |
| "return" |
| |
| # If we don't sleep here 'quit' command races with disk I/O |
| sleep 0.5 |
| |
| # Ignore the JOB_STATUS_CHANGE events while shutting down the VM. Depending on |
| # the timing, jobs may or may not transition through a paused state. |
| _send_qemu_cmd $h "{ 'execute': 'quit' }" "return" |
| wait=1 _cleanup_qemu | grep -v 'JOB_STATUS_CHANGE' |
| |
| echo |
| echo === Start active commit job and exit qemu === |
| echo |
| |
| _launch_qemu \ |
| -drive file="${TEST_IMG}",cache=$CACHEMODE,aio=$AIOMODE,driver=$IMGFMT,id=disk |
| h=$QEMU_HANDLE |
| _send_qemu_cmd $h "{ 'execute': 'qmp_capabilities' }" 'return' |
| |
| _send_qemu_cmd $h \ |
| "{ 'execute': 'block-commit', |
| 'arguments': { 'device': 'disk', |
| 'base':'$TEST_IMG.base', |
| 'speed': 65536 } }" \ |
| "return" |
| |
| # If we don't sleep here 'quit' command races with disk I/O |
| sleep 0.5 |
| |
| _send_qemu_cmd $h "{ 'execute': 'quit' }" "return" |
| wait=1 _cleanup_qemu | grep -v 'JOB_STATUS_CHANGE' |
| |
| echo |
| echo === Start mirror job and exit qemu === |
| echo |
| |
| _launch_qemu \ |
| -drive file="${TEST_IMG}",cache=$CACHEMODE,aio=$AIOMODE,driver=$IMGFMT,id=disk |
| h=$QEMU_HANDLE |
| _send_qemu_cmd $h "{ 'execute': 'qmp_capabilities' }" 'return' |
| |
| _send_qemu_cmd $h \ |
| "{ 'execute': 'drive-mirror', |
| 'arguments': { 'device': 'disk', |
| 'target': '$TEST_IMG.copy', |
| 'format': '$IMGFMT', |
| 'sync': 'full', |
| 'speed': 65536 } }" \ |
| "return" |
| |
| # If we don't sleep here 'quit' command may be handled before |
| # the first mirror iteration is done |
| sleep 0.5 |
| |
| _send_qemu_cmd $h "{ 'execute': 'quit' }" "return" |
| wait=1 _cleanup_qemu | grep -v 'JOB_STATUS_CHANGE' |
| |
| echo |
| echo === Start backup job and exit qemu === |
| echo |
| |
| _launch_qemu \ |
| -drive file="${TEST_IMG}",cache=$CACHEMODE,aio=$AIOMODE,driver=$IMGFMT,id=disk |
| h=$QEMU_HANDLE |
| _send_qemu_cmd $h "{ 'execute': 'qmp_capabilities' }" 'return' |
| |
| _send_qemu_cmd $h \ |
| "{ 'execute': 'drive-backup', |
| 'arguments': { 'device': 'disk', |
| 'target': '$TEST_IMG.copy', |
| 'format': '$IMGFMT', |
| 'sync': 'full', |
| 'speed': 65536, |
| 'x-perf': {'max-chunk': 65536} } }" \ |
| "return" |
| |
| # If we don't sleep here 'quit' command races with disk I/O |
| sleep 0.5 |
| |
| _send_qemu_cmd $h "{ 'execute': 'quit' }" "return" |
| wait=1 _cleanup_qemu | grep -v 'JOB_STATUS_CHANGE' |
| |
| echo |
| echo === Start streaming job and exit qemu === |
| echo |
| |
| _launch_qemu \ |
| -drive file="${TEST_IMG}",cache=$CACHEMODE,aio=$AIOMODE,driver=$IMGFMT,id=disk |
| h=$QEMU_HANDLE |
| _send_qemu_cmd $h "{ 'execute': 'qmp_capabilities' }" 'return' |
| |
| _send_qemu_cmd $h \ |
| "{ 'execute': 'block-stream', |
| 'arguments': { 'device': 'disk', |
| 'speed': 65536 } }" \ |
| "return" |
| |
| # If we don't sleep here 'quit' command races with disk I/O |
| sleep 0.5 |
| |
| _send_qemu_cmd $h "{ 'execute': 'quit' }" "return" |
| wait=1 _cleanup_qemu | grep -v 'JOB_STATUS_CHANGE' |
| |
| _check_test_img |
| |
| echo |
| echo === Start mirror to throttled QSD and exit qemu === |
| echo |
| |
| # Mirror to a throttled QSD instance (so that qemu cannot drain the |
| # throttling), wait for READY, then write some data to the device, |
| # and then quit qemu. |
| # (qemu should force-cancel the job and not wait for the data to be |
| # written to the target.) |
| |
| _make_test_img $size |
| |
| # Will be used by this and the next case |
| set_up_throttled_qsd() { |
| $QSD \ |
| --object throttle-group,id=thrgr,limits.bps-total=1048576 \ |
| --blockdev null-co,node-name=null,size=$size \ |
| --blockdev throttle,node-name=throttled,throttle-group=thrgr,file=null \ |
| --nbd-server addr.type=unix,addr.path="$SOCK_DIR/qsd.sock" \ |
| --export nbd,id=exp,node-name=throttled,name=target,writable=true \ |
| --pidfile "$TEST_DIR/qsd.pid" \ |
| --daemonize |
| } |
| |
| set_up_throttled_qsd |
| |
| # Need a virtio-blk device so that qemu-io writes will not block the monitor |
| _launch_qemu \ |
| --blockdev file,node-name=source-proto,filename="$TEST_IMG" \ |
| --blockdev qcow2,node-name=source-fmt,file=source-proto \ |
| --device virtio-blk,id=vblk,drive=source-fmt \ |
| --blockdev "{\"driver\": \"nbd\", |
| \"node-name\": \"target\", |
| \"server\": { |
| \"type\": \"unix\", |
| \"path\": \"$SOCK_DIR/qsd.sock\" |
| }, |
| \"export\": \"target\"}" |
| |
| h=$QEMU_HANDLE |
| _send_qemu_cmd $h '{"execute": "qmp_capabilities"}' 'return' |
| |
| # Use sync=top, so the first pass will not copy the whole image |
| _send_qemu_cmd $h \ |
| '{"execute": "blockdev-mirror", |
| "arguments": { |
| "job-id": "mirror", |
| "device": "source-fmt", |
| "target": "target", |
| "sync": "top" |
| }}' \ |
| 'return' \ |
| | grep -v JOB_STATUS_CHANGE # Ignore these events during creation |
| |
| # This too will be used by this and the next case |
| # $1: QEMU handle |
| # $2: Image size |
| wait_for_job_and_quit() { |
| h=$1 |
| size=$2 |
| |
| # List of expected events |
| capture_events='BLOCK_JOB_READY JOB_STATUS_CHANGE' |
| _wait_event $h 'BLOCK_JOB_READY' |
| QEMU_EVENTS= # Ignore all JOB_STATUS_CHANGE events that came before READY |
| |
| # Write something to the device for post-READY mirroring. Write it in |
| # blocks matching the cluster size, each spaced one block apart, so |
| # that the mirror job will have to spawn one request per cluster. |
| # Because the number of concurrent requests is limited (to 16), this |
| # limits the number of bytes concurrently in flight, which speeds up |
| # cancelling the job (in-flight requests still are waited for). |
| # To limit the number of bytes in flight, we could alternatively pass |
| # something for blockdev-mirror's @buf-size parameter, but |
| # block-commit does not have such a parameter, so we need to figure |
| # something out that works for both. |
| |
| cluster_size=65536 |
| step=$((cluster_size * 2)) |
| |
| echo '--- Writing data to the virtio-blk device ---' |
| |
| for ofs in $(seq 0 $step $((size - step))); do |
| qemu_io_cmd="qemu-io -d vblk/virtio-backend " |
| qemu_io_cmd+="\\\"aio_write $ofs $cluster_size\\\"" |
| |
| # Do not include these requests in the reference output |
| # (it's just too much) |
| silent=yes _send_qemu_cmd $h \ |
| "{\"execute\": \"human-monitor-command\", |
| \"arguments\": { |
| \"command-line\": \"$qemu_io_cmd\" |
| }}" \ |
| 'return' |
| done |
| |
| # Wait until the job's length is updated to reflect the write requests |
| |
| # We have written to half of the device, so this is the expected job length |
| final_len=$((size / 2)) |
| timeout=100 # unit: 0.1 seconds |
| while true; do |
| len=$( |
| _send_qemu_cmd $h \ |
| '{"execute": "query-block-jobs"}' \ |
| 'return.*"len": [0-9]\+' \ |
| | grep 'return.*"len": [0-9]\+' \ |
| | sed -e 's/.*"len": \([0-9]\+\).*/\1/' |
| ) |
| if [ "$len" -eq "$final_len" ]; then |
| break |
| fi |
| timeout=$((timeout - 1)) |
| if [ "$timeout" -eq 0 ]; then |
| echo "ERROR: Timeout waiting for job to reach len=$final_len" |
| break |
| fi |
| sleep 0.1 |
| done |
| |
| sleep 1 |
| |
| _send_qemu_cmd $h \ |
| '{"execute": "quit"}' \ |
| 'return' |
| |
| # List of expected events |
| capture_events='BLOCK_JOB_CANCELLED JOB_STATUS_CHANGE SHUTDOWN' |
| _wait_event $h 'SHUTDOWN' |
| QEMU_EVENTS= # Ignore all JOB_STATUS_CHANGE events that came before SHUTDOWN |
| _wait_event $h 'JOB_STATUS_CHANGE' # standby |
| _wait_event $h 'JOB_STATUS_CHANGE' # ready |
| _wait_event $h 'JOB_STATUS_CHANGE' # aborting |
| # Filter the offset (depends on when exactly `quit` was issued) |
| _wait_event $h 'BLOCK_JOB_CANCELLED' \ |
| | sed -e 's/"offset": [0-9]\+/"offset": (filtered)/' |
| _wait_event $h 'JOB_STATUS_CHANGE' # concluded |
| _wait_event $h 'JOB_STATUS_CHANGE' # null |
| |
| wait=yes _cleanup_qemu |
| |
| kill -SIGTERM "$(cat "$TEST_DIR/qsd.pid")" |
| } |
| |
| wait_for_job_and_quit $h $size |
| |
| echo |
| echo === Start active commit to throttled QSD and exit qemu === |
| echo |
| |
| # Same as the above, but instead of mirroring, do an active commit |
| |
| _make_test_img $size |
| |
| set_up_throttled_qsd |
| |
| _launch_qemu \ |
| --blockdev "{\"driver\": \"nbd\", |
| \"node-name\": \"target\", |
| \"server\": { |
| \"type\": \"unix\", |
| \"path\": \"$SOCK_DIR/qsd.sock\" |
| }, |
| \"export\": \"target\"}" \ |
| --blockdev file,node-name=source-proto,filename="$TEST_IMG" \ |
| --blockdev qcow2,node-name=source-fmt,file=source-proto,backing=target \ |
| --device virtio-blk,id=vblk,drive=source-fmt |
| |
| h=$QEMU_HANDLE |
| _send_qemu_cmd $h '{"execute": "qmp_capabilities"}' 'return' |
| |
| _send_qemu_cmd $h \ |
| '{"execute": "block-commit", |
| "arguments": { |
| "job-id": "commit", |
| "device": "source-fmt" |
| }}' \ |
| 'return' \ |
| | grep -v JOB_STATUS_CHANGE # Ignore these events during creation |
| |
| wait_for_job_and_quit $h $size |
| |
| # success, all done |
| echo "*** done" |
| rm -f $seq.full |
| status=0 |