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 # test that bitmap is still here
107 removed
= (not migrate_bitmaps
) and persistent
108 self
.check_bitmap(self
.vm_a
, False if removed
else sha256
)
110 self
.vm_a
.qmp('cont')
112 # test that bitmap is still here after invalidation
113 self
.check_bitmap(self
.vm_a
, sha256
)
115 # shutdown and check that invalidation didn't fail
118 # catch 'Could not reopen qcow2 layer: Bitmap already exists'
120 log
= self
.vm_a
.get_log()
121 log
= re
.sub(r
'^\[I \d+\.\d+\] OPENED\n', '', log
)
122 log
= re
.sub(r
'^(wrote .* bytes at offset .*\n.*KiB.*ops.*sec.*\n){3}',
124 log
= re
.sub(r
'\[I \+\d+\.\d+\] CLOSED\n?$', '', log
)
125 self
.assertEqual(log
, '')
127 # test that bitmap is still persistent
129 self
.check_bitmap(self
.vm_a
, sha256
if persistent
else False)
131 def do_test_migration(self
, persistent
, migrate_bitmaps
, online
,
135 # regions = ((start, count), ...)
136 regions
= ((0, 0x10000),
140 should_migrate
= migrate_bitmaps
or persistent
and shared_storage
141 mig_caps
= [{'capability': 'events', 'state': True}]
143 mig_caps
.append({'capability': 'dirty-bitmaps', 'state': True})
145 result
= self
.vm_a
.qmp('migrate-set-capabilities',
146 capabilities
=mig_caps
)
147 self
.assert_qmp(result
, 'return', {})
149 self
.vm_b
.add_incoming(incoming_cmd
if online
else "defer")
150 self
.vm_b
.add_drive(disk_a
if shared_storage
else disk_b
)
155 result
= self
.vm_b
.qmp('migrate-set-capabilities',
156 capabilities
=mig_caps
)
157 self
.assert_qmp(result
, 'return', {})
159 self
.add_bitmap(self
.vm_a
, granularity
, persistent
)
161 self
.vm_a
.hmp_qemu_io('drive0', 'write %d %d' % r
)
162 sha256
= self
.get_bitmap_hash(self
.vm_a
)
164 result
= self
.vm_a
.qmp('migrate', uri
=mig_cmd
)
166 event
= self
.vm_a
.event_wait('MIGRATION')
167 if event
['data']['status'] == 'completed':
173 result
= self
.vm_b
.qmp('migrate-set-capabilities',
174 capabilities
=mig_caps
)
175 self
.assert_qmp(result
, 'return', {})
176 result
= self
.vm_b
.qmp('migrate-incoming', uri
=incoming_cmd
)
177 self
.assert_qmp(result
, 'return', {})
180 event
= self
.vm_b
.event_wait('MIGRATION')
181 if event
['data']['status'] == 'completed':
184 self
.check_bitmap(self
.vm_b
, sha256
if should_migrate
else False)
189 # catch 'Could not reopen qcow2 layer: Bitmap already exists'
191 log
= self
.vm_b
.get_log()
192 log
= re
.sub(r
'^\[I \d+\.\d+\] OPENED\n', '', log
)
193 log
= re
.sub(r
'\[I \+\d+\.\d+\] CLOSED\n?$', '', log
)
194 self
.assertEqual(log
, '')
196 # recreate vm_b, as we don't want -incoming option (this will lead
197 # to "cat" process left alive after test finish)
198 self
.vm_b
= iotests
.VM(path_suffix
='b')
199 self
.vm_b
.add_drive(disk_a
if shared_storage
else disk_b
)
201 self
.check_bitmap(self
.vm_b
, sha256
if persistent
else False)
204 def inject_test_case(klass
, name
, method
, *args
, **kwargs
):
205 mc
= operator
.methodcaller(method
, *args
, **kwargs
)
206 setattr(klass
, 'test_' + method
+ name
, lambda self
: mc(self
))
208 for cmb
in list(itertools
.product((True, False), repeat
=4)):
209 name
= ('_' if cmb
[0] else '_not_') + 'persistent_'
210 name
+= ('_' if cmb
[1] else '_not_') + 'migbitmap_'
211 name
+= '_online' if cmb
[2] else '_offline'
212 name
+= '_shared' if cmb
[3] else '_nonshared'
214 inject_test_case(TestDirtyBitmapMigration
, name
, 'do_test_migration',
217 for cmb
in list(itertools
.product((True, False), repeat
=2)):
218 name
= ('_' if cmb
[0] else '_not_') + 'persistent_'
219 name
+= ('_' if cmb
[1] else '_not_') + 'migbitmap'
221 inject_test_case(TestDirtyBitmapMigration
, name
,
222 'do_test_migration_resume_source', *list(cmb
))
224 if __name__
== '__main__':
225 iotests
.main(supported_fmts
=['qcow2'])