10c46a69aSKevin Wolf#!/usr/bin/env python 20b7e7f66SKevin Wolf# 30b7e7f66SKevin Wolf# Test vhdx and file image creation 40b7e7f66SKevin Wolf# 50b7e7f66SKevin Wolf# Copyright (C) 2018 Red Hat, Inc. 60b7e7f66SKevin Wolf# 70c46a69aSKevin Wolf# Creator/Owner: Kevin Wolf <kwolf@redhat.com> 80c46a69aSKevin Wolf# 90b7e7f66SKevin Wolf# This program is free software; you can redistribute it and/or modify 100b7e7f66SKevin Wolf# it under the terms of the GNU General Public License as published by 110b7e7f66SKevin Wolf# the Free Software Foundation; either version 2 of the License, or 120b7e7f66SKevin Wolf# (at your option) any later version. 130b7e7f66SKevin Wolf# 140b7e7f66SKevin Wolf# This program is distributed in the hope that it will be useful, 150b7e7f66SKevin Wolf# but WITHOUT ANY WARRANTY; without even the implied warranty of 160b7e7f66SKevin Wolf# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 170b7e7f66SKevin Wolf# GNU General Public License for more details. 180b7e7f66SKevin Wolf# 190b7e7f66SKevin Wolf# You should have received a copy of the GNU General Public License 200b7e7f66SKevin Wolf# along with this program. If not, see <http://www.gnu.org/licenses/>. 210b7e7f66SKevin Wolf# 220b7e7f66SKevin Wolf 230c46a69aSKevin Wolfimport iotests 240c46a69aSKevin Wolffrom iotests import imgfmt 250b7e7f66SKevin Wolf 260c46a69aSKevin Wolfiotests.verify_image_format(supported_fmts=['vhdx']) 270c46a69aSKevin Wolfiotests.verify_protocol(supported=['file']) 280b7e7f66SKevin Wolf 290c46a69aSKevin Wolfwith iotests.FilePath('t.vhdx') as disk_path, \ 300c46a69aSKevin Wolf iotests.VM() as vm: 310b7e7f66SKevin Wolf 320c46a69aSKevin Wolf # 330c46a69aSKevin Wolf # Successful image creation (defaults) 340c46a69aSKevin Wolf # 350c46a69aSKevin Wolf iotests.log("=== Successful image creation (defaults) ===") 360c46a69aSKevin Wolf iotests.log("") 370b7e7f66SKevin Wolf 380c46a69aSKevin Wolf size = 128 * 1024 * 1024 390b7e7f66SKevin Wolf 400c46a69aSKevin Wolf vm.launch() 41*55689338SKevin Wolf vm.blockdev_create({ 'driver': 'file', 420c46a69aSKevin Wolf 'filename': disk_path, 430c46a69aSKevin Wolf 'size': 0 }) 440b7e7f66SKevin Wolf 450c46a69aSKevin Wolf vm.qmp_log('blockdev-add', driver='file', filename=disk_path, 46250c04f5SMax Reitz node_name='imgfile', filters=[iotests.filter_qmp_testfiles]) 470b7e7f66SKevin Wolf 48*55689338SKevin Wolf vm.blockdev_create({ 'driver': imgfmt, 490c46a69aSKevin Wolf 'file': 'imgfile', 500c46a69aSKevin Wolf 'size': size }) 510c46a69aSKevin Wolf vm.shutdown() 520b7e7f66SKevin Wolf 530c46a69aSKevin Wolf iotests.img_info_log(disk_path) 540b7e7f66SKevin Wolf 550c46a69aSKevin Wolf # 560c46a69aSKevin Wolf # Successful image creation (explicit defaults) 570c46a69aSKevin Wolf # 580c46a69aSKevin Wolf iotests.log("=== Successful image creation (explicit defaults) ===") 590c46a69aSKevin Wolf iotests.log("") 600b7e7f66SKevin Wolf 610b7e7f66SKevin Wolf # Choose a different size to show that we got a new image 620c46a69aSKevin Wolf size = 64 * 1024 * 1024 630b7e7f66SKevin Wolf 640c46a69aSKevin Wolf vm.launch() 65*55689338SKevin Wolf vm.blockdev_create({ 'driver': 'file', 660c46a69aSKevin Wolf 'filename': disk_path, 670c46a69aSKevin Wolf 'size': 0 }) 68*55689338SKevin Wolf vm.blockdev_create({ 'driver': imgfmt, 690c46a69aSKevin Wolf 'file': { 700c46a69aSKevin Wolf 'driver': 'file', 710c46a69aSKevin Wolf 'filename': disk_path, 720b7e7f66SKevin Wolf }, 730c46a69aSKevin Wolf 'size': size, 740c46a69aSKevin Wolf 'log-size': 1048576, 750c46a69aSKevin Wolf 'block-size': 8388608, 760c46a69aSKevin Wolf 'subformat': 'dynamic', 770c46a69aSKevin Wolf 'block-state-zero': True }) 780c46a69aSKevin Wolf vm.shutdown() 790b7e7f66SKevin Wolf 800c46a69aSKevin Wolf iotests.img_info_log(disk_path) 810b7e7f66SKevin Wolf 820c46a69aSKevin Wolf # 830c46a69aSKevin Wolf # Successful image creation (with non-default options) 840c46a69aSKevin Wolf # 850c46a69aSKevin Wolf iotests.log("=== Successful image creation (with non-default options) ===") 860c46a69aSKevin Wolf iotests.log("") 870b7e7f66SKevin Wolf 880b7e7f66SKevin Wolf # Choose a different size to show that we got a new image 890c46a69aSKevin Wolf size = 32 * 1024 * 1024 900b7e7f66SKevin Wolf 910c46a69aSKevin Wolf vm.launch() 92*55689338SKevin Wolf vm.blockdev_create({ 'driver': 'file', 930c46a69aSKevin Wolf 'filename': disk_path, 940c46a69aSKevin Wolf 'size': 0 }) 95*55689338SKevin Wolf vm.blockdev_create({ 'driver': imgfmt, 960c46a69aSKevin Wolf 'file': { 970c46a69aSKevin Wolf 'driver': 'file', 980c46a69aSKevin Wolf 'filename': disk_path, 990b7e7f66SKevin Wolf }, 1000c46a69aSKevin Wolf 'size': size, 1010c46a69aSKevin Wolf 'log-size': 8388608, 1020c46a69aSKevin Wolf 'block-size': 268435456, 1030c46a69aSKevin Wolf 'subformat': 'fixed', 1040c46a69aSKevin Wolf 'block-state-zero': False }) 1050c46a69aSKevin Wolf vm.shutdown() 1060b7e7f66SKevin Wolf 1070c46a69aSKevin Wolf iotests.img_info_log(disk_path) 1080b7e7f66SKevin Wolf 1090c46a69aSKevin Wolf # 1100c46a69aSKevin Wolf # Invalid BlockdevRef 1110c46a69aSKevin Wolf # 1120c46a69aSKevin Wolf iotests.log("=== Invalid BlockdevRef ===") 1130c46a69aSKevin Wolf iotests.log("") 1140b7e7f66SKevin Wolf 1150c46a69aSKevin Wolf vm.launch() 116*55689338SKevin Wolf vm.blockdev_create({ 'driver': imgfmt, 1170c46a69aSKevin Wolf 'file': "this doesn't exist", 1180c46a69aSKevin Wolf 'size': size }) 1190c46a69aSKevin Wolf vm.shutdown() 1200b7e7f66SKevin Wolf 1210c46a69aSKevin Wolf # 1220c46a69aSKevin Wolf # Zero size 1230c46a69aSKevin Wolf # 1240c46a69aSKevin Wolf iotests.log("=== Zero size ===") 1250c46a69aSKevin Wolf iotests.log("") 1260b7e7f66SKevin Wolf 1270c46a69aSKevin Wolf vm.add_blockdev('driver=file,filename=%s,node-name=node0' % (disk_path)) 1280c46a69aSKevin Wolf vm.launch() 129*55689338SKevin Wolf vm.blockdev_create({ 'driver': imgfmt, 1300c46a69aSKevin Wolf 'file': 'node0', 1310c46a69aSKevin Wolf 'size': 0 }) 1320c46a69aSKevin Wolf vm.shutdown() 1330b7e7f66SKevin Wolf 1340c46a69aSKevin Wolf iotests.img_info_log(disk_path) 1350b7e7f66SKevin Wolf 1360c46a69aSKevin Wolf # 1370c46a69aSKevin Wolf # Maximum size 1380c46a69aSKevin Wolf # 1390c46a69aSKevin Wolf iotests.log("=== Maximum size ===") 1400c46a69aSKevin Wolf iotests.log("") 1410b7e7f66SKevin Wolf 1420c46a69aSKevin Wolf vm.launch() 143*55689338SKevin Wolf vm.blockdev_create({ 'driver': imgfmt, 1440c46a69aSKevin Wolf 'file': 'node0', 1450c46a69aSKevin Wolf 'size': 70368744177664 }) 1460c46a69aSKevin Wolf vm.shutdown() 1470b7e7f66SKevin Wolf 1480c46a69aSKevin Wolf iotests.img_info_log(disk_path) 1490b7e7f66SKevin Wolf 1500c46a69aSKevin Wolf # 1510c46a69aSKevin Wolf # Invalid sizes 1520c46a69aSKevin Wolf # 1530b7e7f66SKevin Wolf 1540b7e7f66SKevin Wolf # TODO Negative image sizes aren't handled correctly, but this is a problem 1550c46a69aSKevin Wolf # with QAPI's implementation of the 'size' type and affects other commands 1560c46a69aSKevin Wolf # as well. Once this is fixed, we may want to add a test case here. 1570b7e7f66SKevin Wolf 1580b7e7f66SKevin Wolf # 1. 2^64 - 512 1590b7e7f66SKevin Wolf # 2. 2^63 = 8 EB (qemu-img enforces image sizes less than this) 1600b7e7f66SKevin Wolf # 3. 2^63 - 512 (generally valid, but with the image header the file will 1610b7e7f66SKevin Wolf # exceed 63 bits) 1620b7e7f66SKevin Wolf # 4. 2^46 + 1 (one byte more than maximum image size) 1630b7e7f66SKevin Wolf 1640c46a69aSKevin Wolf iotests.log("=== Invalid sizes ===") 1650c46a69aSKevin Wolf iotests.log("") 1660b7e7f66SKevin Wolf 1670c46a69aSKevin Wolf vm.launch() 1680c46a69aSKevin Wolf for size in [ 18446744073709551104, 9223372036854775808, 1690c46a69aSKevin Wolf 9223372036854775296, 70368744177665 ]: 170*55689338SKevin Wolf vm.blockdev_create({ 'driver': imgfmt, 1710c46a69aSKevin Wolf 'file': 'node0', 1720c46a69aSKevin Wolf 'size': size }) 1730c46a69aSKevin Wolf vm.shutdown() 1740b7e7f66SKevin Wolf 1750c46a69aSKevin Wolf # 1760c46a69aSKevin Wolf # Invalid block size 1770c46a69aSKevin Wolf # 1780c46a69aSKevin Wolf iotests.log("=== Invalid block size ===") 1790c46a69aSKevin Wolf iotests.log("") 1800b7e7f66SKevin Wolf 1810c46a69aSKevin Wolf vm.launch() 1820c46a69aSKevin Wolf for bsize in [ 1234567, 128, 3145728, 536870912, 0 ]: 183*55689338SKevin Wolf vm.blockdev_create({ 'driver': imgfmt, 1840c46a69aSKevin Wolf 'file': 'node0', 1850c46a69aSKevin Wolf 'size': 67108864, 1860c46a69aSKevin Wolf 'block-size': bsize }) 1870c46a69aSKevin Wolf vm.shutdown() 1880b7e7f66SKevin Wolf 1890c46a69aSKevin Wolf # 1900c46a69aSKevin Wolf # Invalid log size 1910c46a69aSKevin Wolf # 1920c46a69aSKevin Wolf iotests.log("=== Invalid log size ===") 1930c46a69aSKevin Wolf iotests.log("") 1940b7e7f66SKevin Wolf 1950c46a69aSKevin Wolf vm.launch() 1960c46a69aSKevin Wolf for lsize in [ 1234567, 128, 4294967296, 0 ]: 197*55689338SKevin Wolf vm.blockdev_create({ 'driver': imgfmt, 1980c46a69aSKevin Wolf 'file': 'node0', 1990c46a69aSKevin Wolf 'size': 67108864, 2000c46a69aSKevin Wolf 'log-size': lsize }) 2010c46a69aSKevin Wolf vm.shutdown() 202