generic: hole punching followed by writes in the same range
authorFilipe Manana <>
Tue, 30 May 2017 04:52:50 +0000 (05:52 +0100)
committerEryu Guan <>
Fri, 2 Jun 2017 03:22:08 +0000 (11:22 +0800)
Test that if we punch a hole in a file, with either a range that goes
beyond the file's size or covers a file range that is already a hole,
and that if after we do some buffered write operations that cover
different parts of the hole, no warnings are emmitted in syslog/dmesg
and the file's content is correct after remounting the filesystem.

This test is motivated by a bug in btrfs that is manifested in kernel
4.12-rc1 onwards (the bug existed long time ago but was not so easy
to expose before 4.12-rc1). The btrfs patch that fixes the issue is
titled: "Btrfs: fix invalid extent maps due to hole punching".

Signed-off-by: Filipe Manana <>
Reviewed-by: Eryu Guan <>
Signed-off-by: Eryu Guan <>
tests/generic/439 [new file with mode: 0755]
tests/generic/439.out [new file with mode: 0644]

diff --git a/tests/generic/439 b/tests/generic/439
new file mode 100755 (executable)
index 0000000..956053f
--- /dev/null
@@ -0,0 +1,80 @@
+#! /bin/bash
+# FS QA Test No. generic/439
+# Test that if we punch a hole in a file, with either a range that goes beyond
+# the file's size or covers a file range that is already a hole, and that if
+# after we do some buffered write operations that cover different parts of the
+# hole, no warnings are emmitted in syslog/dmesg and the file's content is
+# correct after remounting the filesystem.
+# Copyright (C) 2017 SUSE Linux Products GmbH. All Rights Reserved.
+# Author: Filipe Manana <>
+# 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.
+# This program is distributed in the hope that it would be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# 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, write the Free Software Foundation,
+# Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+seq=`basename $0`
+echo "QA output created by $seq"
+status=1       # failure is the default!
+trap "_cleanup; exit \$status" 0 1 2 3 15
+       cd /
+       rm -f $tmp.*
+# get standard environment, filters and checks
+. ./common/rc
+. ./common/filter
+# real QA test starts here
+_supported_fs generic
+_supported_os Linux
+_require_xfs_io_command "fpunch"
+rm -f $seqres.full
+_scratch_mkfs >>$seqres.full 2>&1
+$XFS_IO_PROG -f -c "pwrite -S 0xaa 0 100K" $SCRATCH_MNT/f | _filter_xfs_io
+$XFS_IO_PROG -c "fpunch 60K 90K" $SCRATCH_MNT/f
+$XFS_IO_PROG -c "pwrite -S 0xbb -b 100K 50K 100K" $SCRATCH_MNT/f | _filter_xfs_io
+$XFS_IO_PROG -c "pwrite -S 0xcc -b 50K 100K 50K" $SCRATCH_MNT/f | _filter_xfs_io
+$XFS_IO_PROG -f -c "fpunch 695K 820K" $SCRATCH_MNT/f2
+$XFS_IO_PROG -c "pwrite -S 0xaa 1008K 307K" $SCRATCH_MNT/f2 |  _filter_xfs_io
+$XFS_IO_PROG -c "pwrite -S 0xbb -b 630K 1073K 630K" $SCRATCH_MNT/f2 \
+       | _filter_xfs_io
+$XFS_IO_PROG -c "pwrite -S 0xcc -b 459K 1068K 459K" $SCRATCH_MNT/f2 \
+       | _filter_xfs_io
+echo "File f contents after remounting filesystem:"
+od -t x1 $SCRATCH_MNT/f
+echo "File f2 contents after remounting filesystem:"
+od -t x1 $SCRATCH_MNT/f2
diff --git a/tests/generic/439.out b/tests/generic/439.out
new file mode 100644 (file)
index 0000000..3644641
--- /dev/null
@@ -0,0 +1,31 @@
+QA output created by 439
+wrote 102400/102400 bytes at offset 0
+XXX Bytes, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec)
+wrote 102400/102400 bytes at offset 51200
+XXX Bytes, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec)
+wrote 51200/51200 bytes at offset 102400
+XXX Bytes, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec)
+wrote 314368/314368 bytes at offset 1032192
+XXX Bytes, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec)
+wrote 645120/645120 bytes at offset 1098752
+XXX Bytes, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec)
+wrote 470016/470016 bytes at offset 1093632
+XXX Bytes, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec)
+File f contents after remounting filesystem:
+0000000 aa aa aa aa aa aa aa aa aa aa aa aa aa aa aa aa
+0144000 bb bb bb bb bb bb bb bb bb bb bb bb bb bb bb bb
+0310000 cc cc cc cc cc cc cc cc cc cc cc cc cc cc cc cc
+File f2 contents after remounting filesystem:
+0000000 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
+3740000 aa aa aa aa aa aa aa aa aa aa aa aa aa aa aa aa
+4130000 cc cc cc cc cc cc cc cc cc cc cc cc cc cc cc cc
+5756000 bb bb bb bb bb bb bb bb bb bb bb bb bb bb bb bb
index c804b05d413389051a52bd158c5b7b26d42b4cb3..5d3e4dcf732ef24ac87b6288a9a40e7134711a91 100644 (file)
 436 auto quick rw
 437 auto quick
 438 auto
 436 auto quick rw
 437 auto quick
 438 auto
+439 auto quick punch