4 # Tests for drive-backup
6 # Copyright (C) 2013 Red Hat, Inc.
10 # This program is free software; you can redistribute it and/or modify
11 # it under the terms of the GNU General Public License as published by
12 # the Free Software Foundation; either version 2 of the License, or
13 # (at your option) any later version.
15 # This program is distributed in the hope that it will be useful,
16 # but WITHOUT ANY WARRANTY; without even the implied warranty of
17 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 # GNU General Public License for more details.
20 # You should have received a copy of the GNU General Public License
21 # along with this program. If not, see <http://www.gnu.org/licenses/>.
27 from iotests import qemu_img, qemu_io, create_image
29 backing_img = os.path.join(iotests.test_dir, 'backing.img')
30 test_img = os.path.join(iotests.test_dir, 'test.img')
31 target_img = os.path.join(iotests.test_dir, 'target.img')
33 def img_create(img, fmt=iotests.imgfmt, size='64M', **kwargs):
34 fullname = os.path.join(iotests.test_dir, '%s.%s' % (img, fmt))
36 for k,v in kwargs.items():
37 optargs = optargs + ['-o', '%s=%s' % (k,v)]
38 args = ['create', '-f', fmt] + optargs + [fullname, size]
39 iotests.qemu_img(*args)
48 def io_write_patterns(img, patterns):
49 for pattern in patterns:
50 iotests.qemu_io('-c', 'write -P%s %s %s' % pattern, img)
53 class TestSyncModesNoneAndTop(iotests.QMPTestCase):
54 image_len = 64 * 1024 * 1024 # MB
57 create_image(backing_img, TestSyncModesNoneAndTop.image_len)
58 qemu_img('create', '-f', iotests.imgfmt,
59 '-o', 'backing_file=%s' % backing_img, '-F', 'raw', test_img)
60 qemu_io('-c', 'write -P0x41 0 512', test_img)
61 qemu_io('-c', 'write -P0xd5 1M 32k', test_img)
62 qemu_io('-c', 'write -P0xdc 32M 124k', test_img)
63 qemu_io('-c', 'write -P0xdc 67043328 64k', test_img)
64 self.vm = iotests.VM().add_drive(test_img)
70 os.remove(backing_img)
76 def test_complete_top(self):
77 self.assert_no_active_block_jobs()
78 result = self.vm.qmp('drive-backup', device='drive0', sync='top',
79 format=iotests.imgfmt, target=target_img)
80 self.assert_qmp(result, 'return', {})
82 self.wait_until_completed(check_offset=False)
84 self.assert_no_active_block_jobs()
86 self.assertTrue(iotests.compare_images(test_img, target_img),
87 'target image does not match source after backup')
89 def test_cancel_sync_none(self):
90 self.assert_no_active_block_jobs()
92 result = self.vm.qmp('drive-backup', device='drive0',
93 sync='none', target=target_img)
94 self.assert_qmp(result, 'return', {})
96 self.vm.hmp_qemu_io('drive0', 'write -P0x5e 0 512')
97 self.vm.hmp_qemu_io('drive0', 'aio_flush')
98 # Verify that the original contents exist in the target image.
100 event = self.cancel_and_wait()
101 self.assert_qmp(event, 'data/type', 'backup')
105 self.assertEqual(-1, qemu_io('-c', 'read -P0x41 0 512', target_img).find("verification failed"))
107 class TestBeforeWriteNotifier(iotests.QMPTestCase):
109 self.vm = iotests.VM().add_drive_raw("file=blkdebug::null-co://,id=drive0,align=65536,driver=blkdebug")
114 os.remove(target_img)
116 def test_before_write_notifier(self):
117 self.vm.pause_drive("drive0")
118 result = self.vm.qmp('drive-backup', device='drive0',
119 sync='full', target=target_img,
120 format="file", speed=1)
121 self.assert_qmp(result, 'return', {})
122 result = self.vm.qmp('block-job-pause', device="drive0")
123 self.assert_qmp(result, 'return', {})
124 # Speed is low enough that this must be an uncopied range, which will
125 # trigger the before write notifier
126 self.vm.hmp_qemu_io('drive0', 'aio_write -P 1 512512 512')
127 self.vm.resume_drive("drive0")
128 result = self.vm.qmp('block-job-resume', device="drive0")
129 self.assert_qmp(result, 'return', {})
130 event = self.cancel_and_wait()
131 self.assert_qmp(event, 'data/type', 'backup')
133 class BackupTest(iotests.QMPTestCase):
135 self.vm = iotests.VM()
136 self.test_img = img_create('test')
137 self.dest_img = img_create('dest')
138 self.dest_img2 = img_create('dest2')
139 self.ref_img = img_create('ref')
140 self.vm.add_drive(self.test_img)
145 try_remove(self.test_img)
146 try_remove(self.dest_img)
147 try_remove(self.dest_img2)
148 try_remove(self.ref_img)
150 def hmp_io_writes(self, drive, patterns):
151 for pattern in patterns:
152 self.vm.hmp_qemu_io(drive, 'write -P%s %s %s' % pattern)
153 self.vm.hmp_qemu_io(drive, 'flush')
155 def qmp_backup_and_wait(self, cmd='drive-backup', serror=None,
156 aerror=None, **kwargs):
157 if not self.qmp_backup(cmd, serror, **kwargs):
159 return self.qmp_backup_wait(kwargs['device'], aerror)
161 def qmp_backup(self, cmd='drive-backup',
162 error=None, **kwargs):
163 self.assertTrue('device' in kwargs)
164 res = self.vm.qmp(cmd, **kwargs)
166 self.assert_qmp(res, 'error/desc', error)
168 self.assert_qmp(res, 'return', {})
171 def qmp_backup_wait(self, device, error=None):
172 event = self.vm.event_wait(name="BLOCK_JOB_COMPLETED",
173 match={'data': {'device': device}})
174 self.assertNotEqual(event, None)
176 failure = self.dictpath(event, 'data/error')
177 except AssertionError:
179 self.assert_qmp(event, 'data/offset', event['data']['len'])
183 self.assert_qmp(event, 'data/error', qerror)
186 def test_overlapping_writes(self):
187 # Write something to back up
188 self.hmp_io_writes('drive0', [('42', '0M', '2M')])
190 # Create a reference backup
191 self.qmp_backup_and_wait(device='drive0', format=iotests.imgfmt,
192 sync='full', target=self.ref_img,
194 res = self.vm.qmp('block-job-dismiss', id='drive0')
195 self.assert_qmp(res, 'return', {})
197 # Now to the test backup: We simulate the following guest
199 # (1) [1M + 64k, 1M + 128k): Afterwards, everything in that
200 # area should be in the target image, and we must not copy
201 # it again (because the source image has changed now)
202 # (64k is the job's cluster size)
203 # (2) [1M, 2M): The backup job must not get overeager. It
204 # must copy [1M, 1M + 64k) and [1M + 128k, 2M) separately,
205 # but not the area in between.
207 self.qmp_backup(device='drive0', format=iotests.imgfmt, sync='full',
208 target=self.dest_img, speed=1, auto_dismiss=False)
210 self.hmp_io_writes('drive0', [('23', '%ik' % (1024 + 64), '64k'),
213 # Let the job complete
214 res = self.vm.qmp('block-job-set-speed', device='drive0', speed=0)
215 self.assert_qmp(res, 'return', {})
216 self.qmp_backup_wait('drive0')
217 res = self.vm.qmp('block-job-dismiss', id='drive0')
218 self.assert_qmp(res, 'return', {})
220 self.assertTrue(iotests.compare_images(self.ref_img, self.dest_img),
221 'target image does not match reference image')
223 def test_dismiss_false(self):
224 res = self.vm.qmp('query-block-jobs')
225 self.assert_qmp(res, 'return', [])
226 self.qmp_backup_and_wait(device='drive0', format=iotests.imgfmt,
227 sync='full', target=self.dest_img,
229 res = self.vm.qmp('query-block-jobs')
230 self.assert_qmp(res, 'return', [])
232 def test_dismiss_true(self):
233 res = self.vm.qmp('query-block-jobs')
234 self.assert_qmp(res, 'return', [])
235 self.qmp_backup_and_wait(device='drive0', format=iotests.imgfmt,
236 sync='full', target=self.dest_img,
238 res = self.vm.qmp('query-block-jobs')
239 self.assert_qmp(res, 'return[0]/status', 'concluded')
240 res = self.vm.qmp('block-job-dismiss', id='drive0')
241 self.assert_qmp(res, 'return', {})
242 res = self.vm.qmp('query-block-jobs')
243 self.assert_qmp(res, 'return', [])
245 def test_dismiss_bad_id(self):
246 res = self.vm.qmp('query-block-jobs')
247 self.assert_qmp(res, 'return', [])
248 res = self.vm.qmp('block-job-dismiss', id='foobar')
249 self.assert_qmp(res, 'error/class', 'DeviceNotActive')
251 def test_dismiss_collision(self):
252 res = self.vm.qmp('query-block-jobs')
253 self.assert_qmp(res, 'return', [])
254 self.qmp_backup_and_wait(device='drive0', format=iotests.imgfmt,
255 sync='full', target=self.dest_img,
257 res = self.vm.qmp('query-block-jobs')
258 self.assert_qmp(res, 'return[0]/status', 'concluded')
259 # Leave zombie job un-dismissed, observe a failure:
260 res = self.qmp_backup_and_wait(serror="Job ID 'drive0' already in use",
261 device='drive0', format=iotests.imgfmt,
262 sync='full', target=self.dest_img2,
264 self.assertEqual(res, False)
265 # OK, dismiss the zombie.
266 res = self.vm.qmp('block-job-dismiss', id='drive0')
267 self.assert_qmp(res, 'return', {})
268 res = self.vm.qmp('query-block-jobs')
269 self.assert_qmp(res, 'return', [])
270 # Ensure it's really gone.
271 self.qmp_backup_and_wait(device='drive0', format=iotests.imgfmt,
272 sync='full', target=self.dest_img2,
275 def dismissal_failure(self, dismissal_opt):
276 res = self.vm.qmp('query-block-jobs')
277 self.assert_qmp(res, 'return', [])
278 # Give blkdebug something to chew on
279 self.hmp_io_writes('drive0',
281 ('0x55', '8M', '352k'),
282 ('0x78', '15872k', '1M')))
283 # Add destination node via blkdebug
284 res = self.vm.qmp('blockdev-add',
286 driver=iotests.imgfmt,
288 'driver': 'blkdebug',
291 'filename': self.dest_img
294 'event': 'write_aio',
296 'immediately': False,
300 self.assert_qmp(res, 'return', {})
302 res = self.qmp_backup(cmd='blockdev-backup',
303 device='drive0', target='target0',
304 on_target_error='stop',
306 auto_dismiss=dismissal_opt)
308 event = self.vm.event_wait(name="BLOCK_JOB_ERROR",
309 match={'data': {'device': 'drive0'}})
310 self.assertNotEqual(event, None)
311 # OK, job should pause, but it can't do it immediately, as it can't
312 # cancel other parallel requests (which didn't fail)
313 with iotests.Timeout(60, "Timeout waiting for backup actually paused"):
315 res = self.vm.qmp('query-block-jobs')
316 if res['return'][0]['status'] == 'paused':
318 self.assert_qmp(res, 'return[0]/status', 'paused')
319 res = self.vm.qmp('block-job-dismiss', id='drive0')
320 self.assert_qmp(res, 'error/desc',
321 "Job 'drive0' in state 'paused' cannot accept"
322 " command verb 'dismiss'")
323 res = self.vm.qmp('query-block-jobs')
324 self.assert_qmp(res, 'return[0]/status', 'paused')
325 # OK, unstick job and move forward.
326 res = self.vm.qmp('block-job-resume', device='drive0')
327 self.assert_qmp(res, 'return', {})
328 # And now we need to wait for it to conclude;
329 res = self.qmp_backup_wait(device='drive0')
331 if not dismissal_opt:
332 # Job should now be languishing:
333 res = self.vm.qmp('query-block-jobs')
334 self.assert_qmp(res, 'return[0]/status', 'concluded')
335 res = self.vm.qmp('block-job-dismiss', id='drive0')
336 self.assert_qmp(res, 'return', {})
337 res = self.vm.qmp('query-block-jobs')
338 self.assert_qmp(res, 'return', [])
340 def test_dismiss_premature(self):
341 self.dismissal_failure(False)
343 def test_dismiss_erroneous(self):
344 self.dismissal_failure(True)
346 if __name__ == '__main__':
347 iotests.main(supported_fmts=['qcow2', 'qed'],
348 supported_protocols=['file'])