xref: /openbmc/qemu/tests/qemu-iotests/085 (revision 576e1c4c239621482474ba7b495a41bab2d16ae5)
1#!/bin/bash
2#
3# Live snapshot tests
4#
5# This tests live snapshots of images on a running QEMU instance, using
6# QMP commands.  Both single disk snapshots, and transactional group
7# snapshots are performed.
8#
9# Copyright (C) 2014 Red Hat, Inc.
10# Copyright (C) 2015 Igalia, S.L.
11#
12# This program is free software; you can redistribute it and/or modify
13# it under the terms of the GNU General Public License as published by
14# the Free Software Foundation; either version 2 of the License, or
15# (at your option) any later version.
16#
17# This program is distributed in the hope that it will be useful,
18# but WITHOUT ANY WARRANTY; without even the implied warranty of
19# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20# GNU General Public License for more details.
21#
22# You should have received a copy of the GNU General Public License
23# along with this program.  If not, see <http://www.gnu.org/licenses/>.
24#
25
26# creator
27owner=jcody@redhat.com
28
29seq=`basename $0`
30echo "QA output created by $seq"
31
32here=`pwd`
33status=1	# failure is the default!
34
35snapshot_virt0="snapshot-v0.qcow2"
36snapshot_virt1="snapshot-v1.qcow2"
37
38SNAPSHOTS=10
39
40_cleanup()
41{
42    _cleanup_qemu
43    for i in $(seq 1 ${SNAPSHOTS})
44    do
45        rm -f "${TEST_DIR}/${i}-${snapshot_virt0}"
46        rm -f "${TEST_DIR}/${i}-${snapshot_virt1}"
47    done
48    rm -f "${TEST_IMG}" "${TEST_IMG}.1" "${TEST_IMG}.2" "${TEST_IMG}.base"
49
50}
51trap "_cleanup; exit \$status" 0 1 2 3 15
52
53# get standard environment, filters and checks
54. ./common.rc
55. ./common.filter
56. ./common.qemu
57
58_supported_fmt qcow2
59_supported_proto file
60_supported_os Linux
61
62
63# ${1}: unique identifier for the snapshot filename
64function create_single_snapshot()
65{
66    cmd="{ 'execute': 'blockdev-snapshot-sync',
67                      'arguments': { 'device': 'virtio0',
68                                     'snapshot-file':'${TEST_DIR}/${1}-${snapshot_virt0}',
69                                     'format': 'qcow2' } }"
70    _send_qemu_cmd $h "${cmd}" "return"
71}
72
73# ${1}: unique identifier for the snapshot filename
74function create_group_snapshot()
75{
76    cmd="{ 'execute': 'transaction', 'arguments':
77           {'actions': [
78               { 'type': 'blockdev-snapshot-sync', 'data' :
79                   { 'device': 'virtio0',
80                      'snapshot-file': '${TEST_DIR}/${1}-${snapshot_virt0}' } },
81               { 'type': 'blockdev-snapshot-sync', 'data' :
82                   { 'device': 'virtio1',
83                       'snapshot-file': '${TEST_DIR}/${1}-${snapshot_virt1}' } } ]
84             } }"
85
86    _send_qemu_cmd $h "${cmd}" "return"
87}
88
89# ${1}: unique identifier for the snapshot filename
90# ${2}: extra_params to the blockdev-add command
91# ${3}: filename
92function do_blockdev_add()
93{
94    cmd="{ 'execute': 'blockdev-add', 'arguments':
95           { 'driver': 'qcow2', 'node-name': 'snap_${1}', ${2}
96             'file':
97             { 'driver': 'file', 'filename': '${3}',
98               'node-name': 'file_${1}' } } }"
99    _send_qemu_cmd $h "${cmd}" "return"
100}
101
102# ${1}: unique identifier for the snapshot filename
103function add_snapshot_image()
104{
105    base_image="${TEST_DIR}/$((${1}-1))-${snapshot_virt0}"
106    snapshot_file="${TEST_DIR}/${1}-${snapshot_virt0}"
107    _make_test_img -u -b "${base_image}" "$size"
108    mv "${TEST_IMG}" "${snapshot_file}"
109    do_blockdev_add "$1" "'backing': null, " "${snapshot_file}"
110}
111
112# ${1}: unique identifier for the snapshot filename
113# ${2}: expected response, defaults to 'return'
114function blockdev_snapshot()
115{
116    cmd="{ 'execute': 'blockdev-snapshot',
117                      'arguments': { 'node': 'virtio0',
118                                     'overlay':'snap_${1}' } }"
119    _send_qemu_cmd $h "${cmd}" "${2:-return}"
120}
121
122size=128M
123
124_make_test_img $size
125mv "${TEST_IMG}" "${TEST_IMG}.1"
126_make_test_img $size
127mv "${TEST_IMG}" "${TEST_IMG}.2"
128
129echo
130echo === Running QEMU ===
131echo
132
133qemu_comm_method="qmp"
134_launch_qemu -drive file="${TEST_IMG}.1",if=virtio -drive file="${TEST_IMG}.2",if=virtio
135h=$QEMU_HANDLE
136
137echo
138echo === Sending capabilities ===
139echo
140
141_send_qemu_cmd $h "{ 'execute': 'qmp_capabilities' }" "return"
142
143# Tests for the blockdev-snapshot-sync command
144
145echo
146echo === Create a single snapshot on virtio0 ===
147echo
148
149create_single_snapshot 1
150
151
152echo
153echo === Invalid command - missing device and nodename ===
154echo
155
156_send_qemu_cmd $h "{ 'execute': 'blockdev-snapshot-sync',
157                         'arguments': { 'snapshot-file':'${TEST_DIR}/1-${snapshot_virt0}',
158                                     'format': 'qcow2' } }" "error"
159
160echo
161echo === Invalid command - missing snapshot-file ===
162echo
163
164_send_qemu_cmd $h "{ 'execute': 'blockdev-snapshot-sync',
165                         'arguments': { 'device': 'virtio0',
166                                     'format': 'qcow2' } }" "error"
167echo
168echo
169echo === Create several transactional group snapshots ===
170echo
171
172for i in $(seq 2 ${SNAPSHOTS})
173do
174    create_group_snapshot ${i}
175done
176
177# Tests for the blockdev-snapshot command
178
179echo
180echo === Create a couple of snapshots using blockdev-snapshot ===
181echo
182
183SNAPSHOTS=$((${SNAPSHOTS}+1))
184add_snapshot_image ${SNAPSHOTS}
185blockdev_snapshot ${SNAPSHOTS}
186
187SNAPSHOTS=$((${SNAPSHOTS}+1))
188add_snapshot_image ${SNAPSHOTS}
189blockdev_snapshot ${SNAPSHOTS}
190
191echo
192echo === Invalid command - cannot create a snapshot using a file BDS ===
193echo
194
195_send_qemu_cmd $h "{ 'execute': 'blockdev-snapshot',
196                     'arguments': { 'node':'virtio0',
197                                    'overlay':'file_${SNAPSHOTS}' }
198                   }" "error"
199
200echo
201echo === Invalid command - snapshot node used as active layer ===
202echo
203
204blockdev_snapshot ${SNAPSHOTS} error
205
206_send_qemu_cmd $h "{ 'execute': 'blockdev-snapshot',
207                     'arguments': { 'node':'virtio0',
208                                    'overlay':'virtio0' }
209                   }" "error"
210
211_send_qemu_cmd $h "{ 'execute': 'blockdev-snapshot',
212                     'arguments': { 'node':'virtio0',
213                                    'overlay':'virtio1' }
214                   }" "error"
215
216echo
217echo === Invalid command - snapshot node used as backing hd ===
218echo
219
220blockdev_snapshot $((${SNAPSHOTS}-1)) error
221
222echo
223echo === Invalid command - snapshot node has a backing image ===
224echo
225
226SNAPSHOTS=$((${SNAPSHOTS}+1))
227
228TEST_IMG="$TEST_IMG.base" _make_test_img "$size"
229_make_test_img -b "${TEST_IMG}.base" "$size"
230do_blockdev_add ${SNAPSHOTS} "" "${TEST_IMG}"
231blockdev_snapshot ${SNAPSHOTS} error
232
233echo
234echo === Invalid command - The node does not exist ===
235echo
236
237blockdev_snapshot $((${SNAPSHOTS}+1)) error
238
239_send_qemu_cmd $h "{ 'execute': 'blockdev-snapshot',
240                     'arguments': { 'node':'nodevice',
241                                    'overlay':'snap_${SNAPSHOTS}' }
242                   }" "error"
243
244# success, all done
245echo "*** done"
246rm -f $seq.full
247status=0
248