xref: /qemu/tests/qemu-iotests/298 (revision b21e2380)
1#!/usr/bin/env python3
2#
3# Test for preallocate filter
4#
5# Copyright (c) 2020 Virtuozzo International GmbH.
6#
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.
11#
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.
16#
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/>.
19#
20
21import os
22import iotests
23
24MiB = 1024 * 1024
25disk = os.path.join(iotests.test_dir, 'disk')
26overlay = os.path.join(iotests.test_dir, 'overlay')
27refdisk = os.path.join(iotests.test_dir, 'refdisk')
28drive_opts = f'node-name=disk,driver={iotests.imgfmt},' \
29    f'file.node-name=filter,file.driver=preallocate,' \
30    f'file.file.node-name=file,file.file.filename={disk}'
31
32
33class TestPreallocateBase(iotests.QMPTestCase):
34    def setUp(self):
35        iotests.qemu_img_create('-f', iotests.imgfmt, disk, str(10 * MiB))
36
37    def tearDown(self):
38        try:
39            self.check_small()
40            check = iotests.qemu_img_check(disk)
41            self.assertFalse('leaks' in check)
42            self.assertFalse('corruptions' in check)
43            self.assertEqual(check['check-errors'], 0)
44        finally:
45            os.remove(disk)
46
47    def check_big(self):
48        self.assertTrue(os.path.getsize(disk) > 100 * MiB)
49
50    def check_small(self):
51        self.assertTrue(os.path.getsize(disk) < 10 * MiB)
52
53
54class TestQemuImg(TestPreallocateBase):
55    def test_qemu_img(self):
56        p = iotests.QemuIoInteractive('--image-opts', drive_opts)
57
58        p.cmd('write 0 1M')
59        p.cmd('flush')
60
61        self.check_big()
62
63        p.close()
64
65
66class TestPreallocateFilter(TestPreallocateBase):
67    def setUp(self):
68        super().setUp()
69        self.vm = iotests.VM().add_drive(path=None, opts=drive_opts)
70        self.vm.launch()
71
72    def tearDown(self):
73        self.vm.shutdown()
74        super().tearDown()
75
76    def test_prealloc(self):
77        self.vm.hmp_qemu_io('drive0', 'write 0 1M')
78        self.check_big()
79
80    def test_external_snapshot(self):
81        self.test_prealloc()
82
83        result = self.vm.qmp('blockdev-snapshot-sync', node_name='disk',
84                             snapshot_file=overlay,
85                             snapshot_node_name='overlay')
86        self.assert_qmp(result, 'return', {})
87
88        # on reopen to  r-o base preallocation should be dropped
89        self.check_small()
90
91        self.vm.hmp_qemu_io('drive0', 'write 1M 1M')
92
93        result = self.vm.qmp('block-commit', device='overlay')
94        self.assert_qmp(result, 'return', {})
95        self.complete_and_wait()
96
97        # commit of new megabyte should trigger preallocation
98        self.check_big()
99
100    def test_reopen_opts(self):
101        result = self.vm.qmp('blockdev-reopen', options=[{
102            'node-name': 'disk',
103            'driver': iotests.imgfmt,
104            'file': {
105                'node-name': 'filter',
106                'driver': 'preallocate',
107                'prealloc-size': 20 * MiB,
108                'prealloc-align': 5 * MiB,
109                'file': {
110                    'node-name': 'file',
111                    'driver': 'file',
112                    'filename': disk
113                }
114            }
115        }])
116        self.assert_qmp(result, 'return', {})
117
118        self.vm.hmp_qemu_io('drive0', 'write 0 1M')
119        self.assertTrue(os.path.getsize(disk) == 25 * MiB)
120
121
122class TestTruncate(iotests.QMPTestCase):
123    def setUp(self):
124        iotests.qemu_img_create('-f', iotests.imgfmt, disk, str(10 * MiB))
125        iotests.qemu_img_create('-f', iotests.imgfmt, refdisk, str(10 * MiB))
126
127    def tearDown(self):
128        os.remove(disk)
129        os.remove(refdisk)
130
131    def do_test(self, prealloc_mode, new_size):
132        ret = iotests.qemu_io_silent('--image-opts', '-c', 'write 0 10M', '-c',
133                                     f'truncate -m {prealloc_mode} {new_size}',
134                                     drive_opts)
135        self.assertEqual(ret, 0)
136
137        ret = iotests.qemu_io_silent('-f', iotests.imgfmt, '-c', 'write 0 10M',
138                                     '-c',
139                                     f'truncate -m {prealloc_mode} {new_size}',
140                                     refdisk)
141        self.assertEqual(ret, 0)
142
143        stat = os.stat(disk)
144        refstat = os.stat(refdisk)
145
146        # Probably we'll want preallocate filter to keep align to cluster when
147        # shrink preallocation, so, ignore small differece
148        self.assertLess(abs(stat.st_size - refstat.st_size), 64 * 1024)
149
150        # Preallocate filter may leak some internal clusters (for example, if
151        # guest write far over EOF, skipping some clusters - they will remain
152        # fallocated, preallocate filter don't care about such leaks, it drops
153        # only trailing preallocation.
154        self.assertLess(abs(stat.st_blocks - refstat.st_blocks) * 512,
155                        1024 * 1024)
156
157    def test_real_shrink(self):
158        self.do_test('off', '5M')
159
160    def test_truncate_inside_preallocated_area__falloc(self):
161        self.do_test('falloc', '50M')
162
163    def test_truncate_inside_preallocated_area__metadata(self):
164        self.do_test('metadata', '50M')
165
166    def test_truncate_inside_preallocated_area__full(self):
167        self.do_test('full', '50M')
168
169    def test_truncate_inside_preallocated_area__off(self):
170        self.do_test('off', '50M')
171
172    def test_truncate_over_preallocated_area__falloc(self):
173        self.do_test('falloc', '150M')
174
175    def test_truncate_over_preallocated_area__metadata(self):
176        self.do_test('metadata', '150M')
177
178    def test_truncate_over_preallocated_area__full(self):
179        self.do_test('full', '150M')
180
181    def test_truncate_over_preallocated_area__off(self):
182        self.do_test('off', '150M')
183
184
185if __name__ == '__main__':
186    iotests.main(supported_fmts=['qcow2'], required_fmts=['preallocate'])
187