3 # Tests for dirty bitmaps migration.
5 # Copyright (c) 2016-2017 Virtuozzo International GmbH. All rights reserved.
7 # This program is free software; you can redistribute it and/or modify
8 # it under the terms of the GNU General Public License as published by
9 # the Free Software Foundation; either version 2 of the License, or
10 # (at your option) any later version.
12 # This program is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU General Public License for more details.
17 # You should have received a copy of the GNU General Public License
18 # along with this program. If not, see <http://www.gnu.org/licenses/>.
27 from iotests
import qemu_img
30 disk_a
= os
.path
.join(iotests
.test_dir
, 'disk_a')
31 disk_b
= os
.path
.join(iotests
.test_dir
, 'disk_b')
33 mig_file
= os
.path
.join(iotests
.test_dir
, 'mig_file')
34 mig_cmd
= 'exec: cat > ' + mig_file
35 incoming_cmd
= 'exec: cat ' + mig_file
38 class TestDirtyBitmapMigration(iotests
.QMPTestCase
):
47 qemu_img('create', '-f', iotests
.imgfmt
, disk_a
, size
)
48 qemu_img('create', '-f', iotests
.imgfmt
, disk_b
, size
)
50 self
.vm_a
= iotests
.VM(path_suffix
='a').add_drive(disk_a
)
53 self
.vm_b
= iotests
.VM(path_suffix
='b')
55 def add_bitmap(self
, vm
, granularity
, persistent
):
56 params
= {'node': 'drive0',
58 'granularity': granularity
}
60 params
['persistent'] = True
62 result
= vm
.qmp('block-dirty-bitmap-add', **params
)
63 self
.assert_qmp(result
, 'return', {});
65 def get_bitmap_hash(self
, vm
):
66 result
= vm
.qmp('x-debug-block-dirty-bitmap-sha256',
67 node
='drive0', name
='bitmap0')
68 return result
['return']['sha256']
70 def check_bitmap(self
, vm
, sha256
):
71 result
= vm
.qmp('x-debug-block-dirty-bitmap-sha256',
72 node
='drive0', name
='bitmap0')
74 self
.assert_qmp(result
, 'return/sha256', sha256
);
76 self
.assert_qmp(result
, 'error/desc',
77 "Dirty bitmap 'bitmap0' not found");
79 def do_test_migration_resume_source(self
, persistent
, migrate_bitmaps
):
82 # regions = ((start, count), ...)
83 regions
= ((0, 0x10000),
87 mig_caps
= [{'capability': 'events', 'state': True}]
89 mig_caps
.append({'capability': 'dirty-bitmaps', 'state': True})
91 result
= self
.vm_a
.qmp('migrate-set-capabilities',
92 capabilities
=mig_caps
)
93 self
.assert_qmp(result
, 'return', {})
95 self
.add_bitmap(self
.vm_a
, granularity
, persistent
)
97 self
.vm_a
.hmp_qemu_io('drive0', 'write %d %d' % r
)
98 sha256
= self
.get_bitmap_hash(self
.vm_a
)
100 result
= self
.vm_a
.qmp('migrate', uri
=mig_cmd
)
102 event
= self
.vm_a
.event_wait('MIGRATION')
103 if event
['data']['status'] == 'completed':
106 result
= self
.vm_a
.qmp('query-status')
107 if (result
['return']['status'] == 'postmigrate'):
110 # test that bitmap is still here
111 removed
= (not migrate_bitmaps
) and persistent
112 self
.check_bitmap(self
.vm_a
, False if removed
else sha256
)
114 result
= self
.vm_a
.qmp('cont')
115 self
.assert_qmp(result
, 'return', {})
117 # test that bitmap is still here after invalidation
118 self
.check_bitmap(self
.vm_a
, sha256
)
120 # shutdown and check that invalidation didn't fail
123 # catch 'Could not reopen qcow2 layer: Bitmap already exists'
125 log
= self
.vm_a
.get_log()
126 log
= re
.sub(r
'^\[I \d+\.\d+\] OPENED\n', '', log
)
127 log
= re
.sub(r
'^(wrote .* bytes at offset .*\n.*KiB.*ops.*sec.*\n){3}',
129 log
= re
.sub(r
'\[I \+\d+\.\d+\] CLOSED\n?$', '', log
)
130 self
.assertEqual(log
, '')
132 # test that bitmap is still persistent
134 self
.check_bitmap(self
.vm_a
, sha256
if persistent
else False)
136 def do_test_migration(self
, persistent
, migrate_bitmaps
, online
,
140 # regions = ((start, count), ...)
141 regions
= ((0, 0x10000),
145 should_migrate
= migrate_bitmaps
or persistent
and shared_storage
146 mig_caps
= [{'capability': 'events', 'state': True}]
148 mig_caps
.append({'capability': 'dirty-bitmaps', 'state': True})
150 result
= self
.vm_a
.qmp('migrate-set-capabilities',
151 capabilities
=mig_caps
)
152 self
.assert_qmp(result
, 'return', {})
154 self
.vm_b
.add_incoming(incoming_cmd
if online
else "defer")
155 self
.vm_b
.add_drive(disk_a
if shared_storage
else disk_b
)
160 result
= self
.vm_b
.qmp('migrate-set-capabilities',
161 capabilities
=mig_caps
)
162 self
.assert_qmp(result
, 'return', {})
164 self
.add_bitmap(self
.vm_a
, granularity
, persistent
)
166 self
.vm_a
.hmp_qemu_io('drive0', 'write %d %d' % r
)
167 sha256
= self
.get_bitmap_hash(self
.vm_a
)
169 result
= self
.vm_a
.qmp('migrate', uri
=mig_cmd
)
171 event
= self
.vm_a
.event_wait('MIGRATION')
172 if event
['data']['status'] == 'completed':
178 result
= self
.vm_b
.qmp('migrate-set-capabilities',
179 capabilities
=mig_caps
)
180 self
.assert_qmp(result
, 'return', {})
181 result
= self
.vm_b
.qmp('migrate-incoming', uri
=incoming_cmd
)
182 self
.assert_qmp(result
, 'return', {})
185 event
= self
.vm_b
.event_wait('MIGRATION')
186 if event
['data']['status'] == 'completed':
189 self
.check_bitmap(self
.vm_b
, sha256
if should_migrate
else False)
194 # catch 'Could not reopen qcow2 layer: Bitmap already exists'
196 log
= self
.vm_b
.get_log()
197 log
= re
.sub(r
'^\[I \d+\.\d+\] OPENED\n', '', log
)
198 log
= re
.sub(r
'\[I \+\d+\.\d+\] CLOSED\n?$', '', log
)
199 self
.assertEqual(log
, '')
201 # recreate vm_b, as we don't want -incoming option (this will lead
202 # to "cat" process left alive after test finish)
203 self
.vm_b
= iotests
.VM(path_suffix
='b')
204 self
.vm_b
.add_drive(disk_a
if shared_storage
else disk_b
)
206 self
.check_bitmap(self
.vm_b
, sha256
if persistent
else False)
209 def inject_test_case(klass
, name
, method
, *args
, **kwargs
):
210 mc
= operator
.methodcaller(method
, *args
, **kwargs
)
211 setattr(klass
, 'test_' + method
+ name
, lambda self
: mc(self
))
213 for cmb
in list(itertools
.product((True, False), repeat
=4)):
214 name
= ('_' if cmb
[0] else '_not_') + 'persistent_'
215 name
+= ('_' if cmb
[1] else '_not_') + 'migbitmap_'
216 name
+= '_online' if cmb
[2] else '_offline'
217 name
+= '_shared' if cmb
[3] else '_nonshared'
219 inject_test_case(TestDirtyBitmapMigration
, name
, 'do_test_migration',
222 for cmb
in list(itertools
.product((True, False), repeat
=2)):
223 name
= ('_' if cmb
[0] else '_not_') + 'persistent_'
224 name
+= ('_' if cmb
[1] else '_not_') + 'migbitmap'
226 inject_test_case(TestDirtyBitmapMigration
, name
,
227 'do_test_migration_resume_source', *list(cmb
))
229 if __name__
== '__main__':
230 iotests
.main(supported_fmts
=['qcow2'],
231 supported_protocols
=['file'])