2019-03-07 22:58:38 +08:00
|
|
|
#!/usr/bin/env bash
|
2014-03-26 20:05:41 +08:00
|
|
|
#
|
|
|
|
# qcow2 format input validation tests
|
|
|
|
#
|
|
|
|
# Copyright (C) 2013 Red Hat, Inc.
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#
|
|
|
|
|
|
|
|
# creator
|
|
|
|
owner=kwolf@redhat.com
|
|
|
|
|
|
|
|
seq=`basename $0`
|
|
|
|
echo "QA output created by $seq"
|
|
|
|
|
|
|
|
status=1 # failure is the default!
|
|
|
|
|
|
|
|
_cleanup()
|
|
|
|
{
|
2019-11-08 00:37:01 +08:00
|
|
|
_rm_test_img "$TEST_IMG.snap"
|
2014-03-26 20:06:06 +08:00
|
|
|
_cleanup_test_img
|
2014-03-26 20:05:41 +08:00
|
|
|
}
|
|
|
|
trap "_cleanup; exit \$status" 0 1 2 3 15
|
|
|
|
|
|
|
|
# get standard environment, filters and checks
|
|
|
|
. ./common.rc
|
|
|
|
. ./common.filter
|
|
|
|
|
|
|
|
_supported_fmt qcow2
|
2014-10-20 19:47:11 +08:00
|
|
|
_supported_proto file
|
2014-03-26 20:05:41 +08:00
|
|
|
_supported_os Linux
|
2019-11-08 00:37:07 +08:00
|
|
|
# - Internal snapshots are (currently) impossible with refcount_bits=1,
|
|
|
|
# and generally impossible with external data files
|
2018-01-18 00:54:20 +08:00
|
|
|
# - This is generally a test for compat=1.1 images
|
2019-11-08 00:37:07 +08:00
|
|
|
_unsupported_imgopts 'refcount_bits=1[^0-9]' data_file 'compat=0.10'
|
2014-03-26 20:05:41 +08:00
|
|
|
|
qcow2: introduce compression type feature
The patch adds some preparation parts for incompatible compression type
feature to qcow2 allowing the use different compression methods for
image clusters (de)compressing.
It is implied that the compression type is set on the image creation and
can be changed only later by image conversion, thus compression type
defines the only compression algorithm used for the image, and thus,
for all image clusters.
The goal of the feature is to add support of other compression methods
to qcow2. For example, ZSTD which is more effective on compression than ZLIB.
The default compression is ZLIB. Images created with ZLIB compression type
are backward compatible with older qemu versions.
Adding of the compression type breaks a number of tests because now the
compression type is reported on image creation and there are some changes
in the qcow2 header in size and offsets.
The tests are fixed in the following ways:
* filter out compression_type for many tests
* fix header size, feature table size and backing file offset
affected tests: 031, 036, 061, 080
header_size +=8: 1 byte compression type
7 bytes padding
feature_table += 48: incompatible feature compression type
backing_file_offset += 56 (8 + 48 -> header_change + feature_table_change)
* add "compression type" for test output matching when it isn't filtered
affected tests: 049, 060, 061, 065, 082, 085, 144, 182, 185, 198, 206,
242, 255, 274, 280
Signed-off-by: Denis Plotnikov <dplotnikov@virtuozzo.com>
Reviewed-by: Vladimir Sementsov-Ogievskiy <vsementsov@virtuozzo.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Max Reitz <mreitz@redhat.com>
QAPI part:
Acked-by: Markus Armbruster <armbru@redhat.com>
Message-Id: <20200507082521.29210-2-dplotnikov@virtuozzo.com>
Signed-off-by: Max Reitz <mreitz@redhat.com>
2020-05-07 16:25:18 +08:00
|
|
|
header_size=112
|
2014-03-26 20:05:42 +08:00
|
|
|
|
|
|
|
offset_backing_file_offset=8
|
2014-03-26 20:05:47 +08:00
|
|
|
offset_backing_file_size=16
|
2014-03-26 20:05:46 +08:00
|
|
|
offset_l1_size=36
|
|
|
|
offset_l1_table_offset=40
|
2014-03-26 20:05:44 +08:00
|
|
|
offset_refcount_table_offset=48
|
2014-03-26 20:05:43 +08:00
|
|
|
offset_refcount_table_clusters=56
|
2014-03-26 20:05:45 +08:00
|
|
|
offset_nb_snapshots=60
|
|
|
|
offset_snapshots_offset=64
|
2014-03-26 20:05:41 +08:00
|
|
|
offset_header_size=100
|
|
|
|
offset_ext_magic=$header_size
|
|
|
|
offset_ext_size=$((header_size + 4))
|
|
|
|
|
2014-03-29 01:06:31 +08:00
|
|
|
offset_l2_table_0=$((0x40000))
|
|
|
|
|
2014-03-26 20:06:06 +08:00
|
|
|
offset_snap1=$((0x70000))
|
|
|
|
offset_snap1_l1_offset=$((offset_snap1 + 0))
|
|
|
|
offset_snap1_l1_size=$((offset_snap1 + 8))
|
|
|
|
|
2014-03-26 20:05:41 +08:00
|
|
|
echo
|
|
|
|
echo "== Huge header size =="
|
|
|
|
_make_test_img 64M
|
|
|
|
poke_file "$TEST_IMG" "$offset_header_size" "\xff\xff\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
poke_file "$TEST_IMG" "$offset_header_size" "\x7f\xff\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
2014-03-26 20:05:42 +08:00
|
|
|
echo
|
|
|
|
echo "== Huge unknown header extension =="
|
|
|
|
_make_test_img 64M
|
|
|
|
poke_file "$TEST_IMG" "$offset_backing_file_offset" "\xff\xff\xff\xff\xff\xff\xff\xff"
|
|
|
|
poke_file "$TEST_IMG" "$offset_ext_magic" "\x12\x34\x56\x78"
|
|
|
|
poke_file "$TEST_IMG" "$offset_ext_size" "\x7f\xff\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
2014-11-26 01:12:40 +08:00
|
|
|
poke_file "$TEST_IMG" "$offset_backing_file_offset" "\x00\x00\x00\x00\x00\x00\x00\x$(printf %x $offset_ext_size)"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
2014-03-26 20:05:42 +08:00
|
|
|
poke_file "$TEST_IMG" "$offset_backing_file_offset" "\x00\x00\x00\x00\x00\x00\x00\x00"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
2014-03-26 20:05:43 +08:00
|
|
|
echo
|
|
|
|
echo "== Huge refcount table size =="
|
|
|
|
_make_test_img 64M
|
|
|
|
poke_file "$TEST_IMG" "$offset_refcount_table_clusters" "\xff\xff\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
poke_file "$TEST_IMG" "$offset_refcount_table_clusters" "\x00\x02\x00\x01"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
2014-03-26 20:05:44 +08:00
|
|
|
echo
|
|
|
|
echo "== Misaligned refcount table =="
|
|
|
|
_make_test_img 64M
|
|
|
|
poke_file "$TEST_IMG" "$offset_refcount_table_offset" "\x12\x34\x56\x78\x90\xab\xcd\xef"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
|
|
|
echo
|
|
|
|
echo "== Huge refcount offset =="
|
|
|
|
_make_test_img 64M
|
|
|
|
poke_file "$TEST_IMG" "$offset_refcount_table_offset" "\xff\xff\xff\xff\xff\xff\x00\x00"
|
|
|
|
poke_file "$TEST_IMG" "$offset_refcount_table_clusters" "\x00\x00\x00\x7f"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
2014-03-26 20:05:43 +08:00
|
|
|
|
2014-03-26 20:05:45 +08:00
|
|
|
echo
|
|
|
|
echo "== Invalid snapshot table =="
|
|
|
|
_make_test_img 64M
|
|
|
|
poke_file "$TEST_IMG" "$offset_nb_snapshots" "\xff\xff\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
poke_file "$TEST_IMG" "$offset_nb_snapshots" "\x7f\xff\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
|
|
|
poke_file "$TEST_IMG" "$offset_snapshots_offset" "\xff\xff\xff\xff\xff\xff\x00\x00"
|
|
|
|
poke_file "$TEST_IMG" "$offset_nb_snapshots" "\x00\x00\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
|
|
|
poke_file "$TEST_IMG" "$offset_snapshots_offset" "\x12\x34\x56\x78\x90\xab\xcd\xef"
|
|
|
|
poke_file "$TEST_IMG" "$offset_nb_snapshots" "\x00\x00\x00\x00"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
|
|
|
echo
|
|
|
|
echo "== Hitting snapshot table size limit =="
|
|
|
|
_make_test_img 64M
|
|
|
|
# Put the refcount table in a more or less safe place (16 MB)
|
|
|
|
poke_file "$TEST_IMG" "$offset_snapshots_offset" "\x00\x00\x00\x00\x01\x00\x00\x00"
|
|
|
|
poke_file "$TEST_IMG" "$offset_nb_snapshots" "\x00\x01\x00\x00"
|
|
|
|
{ $QEMU_IMG snapshot -c test $TEST_IMG; } 2>&1 | _filter_testdir
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
2014-03-26 20:05:46 +08:00
|
|
|
echo
|
|
|
|
echo "== Invalid L1 table =="
|
|
|
|
_make_test_img 64M
|
|
|
|
poke_file "$TEST_IMG" "$offset_l1_size" "\xff\xff\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
poke_file "$TEST_IMG" "$offset_l1_size" "\x7f\xff\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
|
|
|
poke_file "$TEST_IMG" "$offset_l1_table_offset" "\x7f\xff\xff\xff\xff\xff\x00\x00"
|
|
|
|
poke_file "$TEST_IMG" "$offset_l1_size" "\x00\x00\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
|
|
|
poke_file "$TEST_IMG" "$offset_l1_table_offset" "\x12\x34\x56\x78\x90\xab\xcd\xef"
|
|
|
|
poke_file "$TEST_IMG" "$offset_l1_size" "\x00\x00\x00\x01"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
2014-03-26 20:06:04 +08:00
|
|
|
echo
|
|
|
|
echo "== Invalid L1 table (with internal snapshot in the image) =="
|
|
|
|
_make_test_img 64M
|
|
|
|
{ $QEMU_IMG snapshot -c foo $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
poke_file "$TEST_IMG" "$offset_l1_size" "\x00\x00\x00\x00"
|
|
|
|
_img_info
|
|
|
|
|
2014-03-26 20:05:47 +08:00
|
|
|
echo
|
|
|
|
echo "== Invalid backing file size =="
|
|
|
|
_make_test_img 64M
|
|
|
|
poke_file "$TEST_IMG" "$offset_backing_file_offset" "\x00\x00\x00\x00\x00\x00\x10\x00"
|
|
|
|
poke_file "$TEST_IMG" "$offset_backing_file_size" "\xff\xff\xff\xff"
|
|
|
|
{ $QEMU_IO -c "read 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
2014-03-29 01:06:31 +08:00
|
|
|
echo
|
|
|
|
echo "== Invalid L2 entry (huge physical offset) =="
|
|
|
|
_make_test_img 64M
|
|
|
|
{ $QEMU_IO -c "write 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
poke_file "$TEST_IMG" "$offset_l2_table_0" "\xbf\xff\xff\xff\xff\xff\x00\x00"
|
|
|
|
{ $QEMU_IMG snapshot -c test $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
poke_file "$TEST_IMG" "$offset_l2_table_0" "\x80\x00\x00\xff\xff\xff\x00\x00"
|
|
|
|
{ $QEMU_IMG snapshot -c test $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
|
2014-03-26 20:06:06 +08:00
|
|
|
echo
|
2018-03-07 00:14:07 +08:00
|
|
|
echo "== Invalid snapshot L1 table offset =="
|
|
|
|
_make_test_img 64M
|
|
|
|
{ $QEMU_IO -c "write 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
{ $QEMU_IMG snapshot -c test $TEST_IMG; } 2>&1 | _filter_testdir
|
|
|
|
poke_file "$TEST_IMG" "$offset_snap1_l1_offset" "\x00\x00\x00\x00\x00\x40\x02\x00"
|
2018-06-06 20:35:51 +08:00
|
|
|
{ $QEMU_IMG convert -l test $TEST_IMG $TEST_IMG.snap; } 2>&1 | _filter_testdir
|
2018-03-07 00:14:08 +08:00
|
|
|
{ $QEMU_IMG amend -o compat=0.10 $TEST_IMG; } 2>&1 | _filter_testdir
|
2018-03-07 00:14:09 +08:00
|
|
|
{ $QEMU_IO -c "open -o overlap-check.inactive-l2=on $TEST_IMG" \
|
|
|
|
-c 'write 0 4k'; } 2>&1 | _filter_qemu_io | _filter_testdir
|
2018-03-07 00:14:10 +08:00
|
|
|
{ $QEMU_IMG snapshot -a test $TEST_IMG; } 2>&1 | _filter_testdir
|
2018-03-07 00:14:11 +08:00
|
|
|
{ $QEMU_IMG snapshot -d test $TEST_IMG; } 2>&1 | _filter_testdir
|
2018-03-07 00:14:12 +08:00
|
|
|
_check_test_img
|
2018-03-07 00:14:07 +08:00
|
|
|
|
|
|
|
echo
|
|
|
|
echo "== Invalid snapshot L1 table size =="
|
2014-03-26 20:06:06 +08:00
|
|
|
_make_test_img 64M
|
|
|
|
{ $QEMU_IO -c "write 0 512" $TEST_IMG; } 2>&1 | _filter_qemu_io | _filter_testdir
|
|
|
|
{ $QEMU_IMG snapshot -c test $TEST_IMG; } 2>&1 | _filter_testdir
|
|
|
|
poke_file "$TEST_IMG" "$offset_snap1_l1_size" "\x10\x00\x00\x00"
|
2018-06-06 20:35:51 +08:00
|
|
|
{ $QEMU_IMG convert -l test $TEST_IMG $TEST_IMG.snap; } 2>&1 | _filter_testdir
|
2018-03-07 00:14:08 +08:00
|
|
|
{ $QEMU_IMG amend -o compat=0.10 $TEST_IMG; } 2>&1 | _filter_testdir
|
2018-03-07 00:14:09 +08:00
|
|
|
{ $QEMU_IO -c "open -o overlap-check.inactive-l2=on $TEST_IMG" \
|
|
|
|
-c 'write 0 4k'; } 2>&1 | _filter_qemu_io | _filter_testdir
|
2018-03-07 00:14:10 +08:00
|
|
|
{ $QEMU_IMG snapshot -a test $TEST_IMG; } 2>&1 | _filter_testdir
|
2018-03-07 00:14:11 +08:00
|
|
|
{ $QEMU_IMG snapshot -d test $TEST_IMG; } 2>&1 | _filter_testdir
|
2018-03-07 00:14:12 +08:00
|
|
|
_check_test_img
|
2014-03-26 20:06:06 +08:00
|
|
|
|
2014-03-26 20:05:41 +08:00
|
|
|
# success, all done
|
|
|
|
echo "*** done"
|
|
|
|
rm -f $seq.full
|
|
|
|
status=0
|