From nobody Fri Oct 24 09:59:48 2025 Delivered-To: importer@patchew.org Received-SPF: pass (zoho.com: domain of gnu.org designates 208.118.235.17 as permitted sender) client-ip=208.118.235.17; envelope-from=qemu-devel-bounces+importer=patchew.org@nongnu.org; helo=lists.gnu.org; Authentication-Results: mx.zohomail.com; dkim=fail; spf=pass (zoho.com: domain of gnu.org designates 208.118.235.17 as permitted sender) smtp.mailfrom=qemu-devel-bounces+importer=patchew.org@nongnu.org Return-Path: Received: from lists.gnu.org (lists.gnu.org [208.118.235.17]) by mx.zohomail.com with SMTPS id 1519660117197401.8178151279385; Mon, 26 Feb 2018 07:48:37 -0800 (PST) Received: from localhost ([::1]:59544 helo=lists.gnu.org) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1eqJux-0006UT-JH for importer@patchew.org; Mon, 26 Feb 2018 09:38:15 -0500 Received: from eggs.gnu.org ([2001:4830:134:3::10]:53014) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1eqJtg-0005qE-0c for qemu-devel@nongnu.org; Mon, 26 Feb 2018 09:36:57 -0500 Received: from Debian-exim by eggs.gnu.org with spam-scanned (Exim 4.71) (envelope-from ) id 1eqJtb-00052p-W5 for qemu-devel@nongnu.org; Mon, 26 Feb 2018 09:36:56 -0500 Received: from fanzine.igalia.com ([91.117.99.155]:33557) by eggs.gnu.org with esmtps (TLS1.0:RSA_AES_128_CBC_SHA1:16) (Exim 4.71) (envelope-from ) id 1eqJtb-00051r-Fp; Mon, 26 Feb 2018 09:36:51 -0500 Received: from [194.100.51.2] (helo=perseus.local) by fanzine.igalia.com with esmtpsa (Cipher TLS1.2:ECDHE_RSA_AES_256_GCM_SHA384:256) (Exim) id 1eqJtY-0005m1-FB; Mon, 26 Feb 2018 15:36:48 +0100 Received: from berto by perseus.local with local (Exim 4.89) (envelope-from ) id 1eqJtF-0003Bl-BS; Mon, 26 Feb 2018 16:36:29 +0200 DKIM-Signature: v=1; a=rsa-sha256; q=dns/txt; c=relaxed/relaxed; d=igalia.com; s=20170329; h=Message-Id:Date:Subject:Cc:To:From; bh=+XaBGjJ/MDmTFDRG5QPXcIponTfWslVkpO9jl2+itC0=; b=KhMxSMemcRVqgVsrn1ER9hOpqtttXJKDczF1YOdmH9/4Lq7gedDDBUAw69gFc0bTtz0Dbok6C1OEcTW9Vp5M77S496EDs4R1HSWCvHBzfCA0ctvI5uKS0yk3tEtrmeg+abWtvXc0AEnjm/PgERvwPOR0PFo+VnOVxHeExFI0bkd1E1HEYq8v2T08RcT5Jp+XZk+ZD44F7T6cgQcTz6AM1a9LeY9t8N9/nkUZJn2VrolCo/0OFJr9VHKNvh3EKlAE+6Z/JjUiIJ0sQJUpobAzxtSfGR7mvWg9YLjSRQ4p/UWfxIZ5/4vI4vSR/AH0ed7FBBRzpMp7bapzcHai2AYm1A==; From: Alberto Garcia To: qemu-devel@nongnu.org Date: Mon, 26 Feb 2018 16:36:23 +0200 Message-Id: <20180226143623.12200-1-berto@igalia.com> X-Mailer: git-send-email 2.11.0 X-detected-operating-system: by eggs.gnu.org: GNU/Linux 2.2.x-3.x (no timestamps) [generic] [fuzzy] X-Received-From: 91.117.99.155 Subject: [Qemu-devel] [PATCH v3] iotests: Test abnormally large size in compressed cluster descriptor X-BeenThere: qemu-devel@nongnu.org X-Mailman-Version: 2.1.21 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Cc: Kevin Wolf , Alberto Garcia , qemu-block@nongnu.org, Max Reitz Errors-To: qemu-devel-bounces+importer=patchew.org@nongnu.org Sender: "Qemu-devel" X-ZohoMail-DKIM: fail (Header signature does not verify) X-ZohoMail: RDKM_2 RSF_0 Z_629925259 SPT_0 Content-Transfer-Encoding: quoted-printable MIME-Version: 1.0 Content-Type: text/plain; charset="utf-8" L2 entries for compressed clusters have a field that indicates the number of sectors used to store the data in the image. That's however not the size of the compressed data itself, just the number of sectors where that data is located. The actual data size is usually not a multiple of the sector size, and therefore cannot be represented with this field. The way it works is that QEMU reads all the specified sectors and starts decompressing the data until there's enough to recover the original uncompressed cluster. If there are any bytes left that haven't been decompressed they are simply ignored. One consequence of this is that even if the size field is larger than it needs to be QEMU can handle it just fine: it will read more data from disk but it will ignore the extra bytes. This test creates an image with two compressed clusters that use 5 sectors (2.5 KB) each, increases the size field to the maximum (8192 sectors, or 4 MB) and verifies that the data can be read without problems. This test is important because while the decompressed data takes exactly one cluster, the maximum value allowed in the compressed size field is twice the cluster size. So although QEMU won't produce images with such large values we need to make sure that it can handle them. Another effect of increasing the size field is that it can make it include data from the following host cluster(s). In this case 'qemu-img check' will detect that the refcounts are not correct, and we'll need to rebuild them. Additionally, this patch also tests that decreasing the size corrupts the image since the original data can no longer be recovered. In this case QEMU returns an error when trying to read the compressed data, but 'qemu-img check' doesn't see anything wrong if the refcounts are consistent. One possible task for the future is to make 'qemu-img check' verify the sizes of the compressed clusters, by trying to decompress the data and checking that the size stored in the L2 entry is correct. Signed-off-by: Alberto Garcia Reviewed-by: Eric Blake --- v3: Add TODO comment, as suggested by Eric. Corrupt the length of the second compressed cluster as well so the uncompressed data would span three host clusters. v2: We now have two scenarios where we make QEMU read data from the next host cluster and from beyond the end of the image. This version also runs qemu-img check on the corrupted image. If the size field is too small, reading fails but qemu-img check succeeds. If the size field is too large, reading succeeds but qemu-img check fails (this can be repaired, though). --- tests/qemu-iotests/122 | 45 ++++++++++++++++++++++++++++++++++++++++++= +++ tests/qemu-iotests/122.out | 31 +++++++++++++++++++++++++++++++ 2 files changed, 76 insertions(+) diff --git a/tests/qemu-iotests/122 b/tests/qemu-iotests/122 index 45b359c2ba..5b9593016c 100755 --- a/tests/qemu-iotests/122 +++ b/tests/qemu-iotests/122 @@ -130,6 +130,51 @@ $QEMU_IO -c "read -P 0 1024k 1022k" "$TEST_IMG" 2>&= 1 | _filter_qemu_io | _fil =20 =20 echo +echo "=3D=3D=3D Corrupted size field in compressed cluster descriptor =3D= =3D=3D" +echo +# Create an empty image, fill half of it with data and compress it. +# The L2 entries of the two compressed clusters are located at +# 0x800000 and 0x800008, their original values are 0x4008000000a00000 +# and 0x4008000000a00802 (5 sectors for compressed data each). +TEST_IMG=3D"$TEST_IMG".1 _make_test_img 8M +$QEMU_IO -c "write -P 0x11 0 4M" "$TEST_IMG".1 2>&1 | _filter_qemu_io | _f= ilter_testdir +$QEMU_IMG convert -c -O qcow2 -o cluster_size=3D2M "$TEST_IMG".1 "$TEST_IM= G" + +# Reduce size of compressed data to 4 sectors: this corrupts the image. +poke_file "$TEST_IMG" $((0x800000)) "\x40\x06" +$QEMU_IO -c "read -P 0x11 0 4M" "$TEST_IMG" 2>&1 | _filter_qemu_io | _fil= ter_testdir + +# 'qemu-img check' however doesn't see anything wrong because it +# doesn't try to decompress the data and the refcounts are consistent. +# TODO: update qemu-img so this can be detected +_check_test_img + +# Increase size of compressed data to the maximum (8192 sectors). +# This makes QEMU read more data (8192 sectors instead of 5, host +# addresses [0xa00000, 0xdfffff]), but the decompression algorithm +# stops once we have enough to restore the uncompressed cluster, so +# the rest of the data is ignored. +poke_file "$TEST_IMG" $((0x800000)) "\x7f\xfe" +# Do it also for the second compressed cluster (L2 entry at 0x800008). +# In this case the compressed data would span 3 host clusters +# (host addresses: [0xa00802, 0xe00801]) +poke_file "$TEST_IMG" $((0x800008)) "\x7f\xfe" + +# Here the image is too small so we're asking QEMU to read beyond the +# end of the image. +$QEMU_IO -c "read -P 0x11 0 4M" "$TEST_IMG" 2>&1 | _filter_qemu_io | _fi= lter_testdir +# But if we grow the image we won't be reading beyond its end anymore. +$QEMU_IO -c "write -P 0x22 4M 4M" "$TEST_IMG" 2>&1 | _filter_qemu_io | _fi= lter_testdir +$QEMU_IO -c "read -P 0x11 0 4M" "$TEST_IMG" 2>&1 | _filter_qemu_io | _fi= lter_testdir + +# The refcount data is however wrong because due to the increased size +# of the compressed data it now reaches the following host clusters. +# This can be repaired by qemu-img check. +_check_test_img -r all +$QEMU_IO -c "read -P 0x11 0 4M" "$TEST_IMG" 2>&1 | _filter_qemu_io | _fi= lter_testdir +$QEMU_IO -c "read -P 0x22 4M 4M" "$TEST_IMG" 2>&1 | _filter_qemu_io | _fi= lter_testdir + +echo echo "=3D=3D=3D Full allocation with -S 0 =3D=3D=3D" echo =20 diff --git a/tests/qemu-iotests/122.out b/tests/qemu-iotests/122.out index 47d8656db8..bdda75650f 100644 --- a/tests/qemu-iotests/122.out +++ b/tests/qemu-iotests/122.out @@ -99,6 +99,37 @@ read 1024/1024 bytes at offset 1047552 read 1046528/1046528 bytes at offset 1048576 1022 KiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec) =20 +=3D=3D=3D Corrupted size field in compressed cluster descriptor =3D=3D=3D + +Formatting 'TEST_DIR/t.IMGFMT.1', fmt=3DIMGFMT size=3D8388608 +wrote 4194304/4194304 bytes at offset 0 +4 MiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec) +read failed: Input/output error +No errors were found on the image. +read 4194304/4194304 bytes at offset 0 +4 MiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec) +wrote 4194304/4194304 bytes at offset 4194304 +4 MiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec) +read 4194304/4194304 bytes at offset 0 +4 MiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec) +ERROR cluster 6 refcount=3D1 reference=3D3 +ERROR cluster 7 refcount=3D1 reference=3D2 +Repairing cluster 6 refcount=3D1 reference=3D3 +Repairing cluster 7 refcount=3D1 reference=3D2 +Repairing OFLAG_COPIED data cluster: l2_entry=3D8000000000c00000 refcount= =3D3 +Repairing OFLAG_COPIED data cluster: l2_entry=3D8000000000e00000 refcount= =3D2 +The following inconsistencies were found and repaired: + + 0 leaked clusters + 4 corruptions + +Double checking the fixed image now... +No errors were found on the image. +read 4194304/4194304 bytes at offset 0 +4 MiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec) +read 4194304/4194304 bytes at offset 4194304 +4 MiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec) + =3D=3D=3D Full allocation with -S 0 =3D=3D=3D =20 Formatting 'TEST_DIR/t.IMGFMT', fmt=3DIMGFMT size=3D67108864 --=20 2.11.0