xfs: test rmapbt updates are correct with insert/collapse range
authorDarrick J. Wong <darrick.wong@oracle.com>
Thu, 31 Aug 2017 14:47:41 +0000 (07:47 -0700)
committerEryu Guan <eguan@redhat.com>
Sun, 3 Sep 2017 04:20:00 +0000 (12:20 +0800)
Make sure that we update the rmapbt correctly when we collapse-range
a file and the extents on both sides of the hole can be merged.  We
can construct this pretty trivially with insert-range and write, so
test that too.

Signed-off-by: Darrick J. Wong <darrick.wong@oracle.com>
Reviewed-by: Eryu Guan <eguan@redhat.com>
Signed-off-by: Eryu Guan <eguan@redhat.com>
src/punch-alternating.c
tests/xfs/114 [new file with mode: 0755]
tests/xfs/114.out [new file with mode: 0644]
tests/xfs/group

index 25228cd5cbed3a75860153543425ca79ba05f446..18dd215197db2b59a79d70734ef4e69394e8b126 100644 (file)
 
 void usage(char *cmd)
 {
-       printf("Usage: %s [-i interval] [-s size] file\n", cmd);
+       printf("Usage: %s [-o offset] [-i interval] [-s size] file\n", cmd);
        printf("Punches every other block in the file by default,\n");
-       printf("or 'size' blocks every 'interval' blocks with options.\n");
+       printf("or 'size' blocks every 'interval' blocks starting at\n");
+       printf("'offset'.  Units are in fstatfs blocks.\n");
        exit(1);
 }
 
@@ -24,6 +25,7 @@ int main(int argc, char *argv[])
        struct stat     s;
        struct statfs   sf;
        off_t           offset;
+       off_t           start_offset = 0;
        int             fd;
        blksize_t       blksz;
        off_t           sz;
@@ -33,11 +35,19 @@ int main(int argc, char *argv[])
        int             size = 1;       /* punch $SIZE blocks ... */
        int             interval = 2;   /* every $INTERVAL blocks */
 
-       while ((c = getopt(argc, argv, "i:s:")) != EOF) {
+       while ((c = getopt(argc, argv, "i:o:s:")) != EOF) {
                switch (c) {
                case 'i':
                        interval = atoi(optarg);
                        break;
+               case 'o':
+                       errno = 0;
+                       start_offset = strtoull(optarg, NULL, 0);
+                       if (errno) {
+                               fprintf(stderr, "invalid offset '%s'\n", optarg);
+                               return 1;
+                       }
+                       break;
                case 's':
                        size = atoi(optarg);
                        break;
@@ -75,7 +85,9 @@ int main(int argc, char *argv[])
        blksz = sf.f_bsize;
 
        mode = FALLOC_FL_PUNCH_HOLE | FALLOC_FL_KEEP_SIZE;
-       for (offset = 0; offset < sz; offset += blksz * interval) {
+       for (offset = start_offset * blksz;
+            offset < sz;
+            offset += blksz * interval) {
                error = fallocate(fd, mode, offset, blksz * size);
                if (error)
                        goto err;
diff --git a/tests/xfs/114 b/tests/xfs/114
new file mode 100755 (executable)
index 0000000..674d884
--- /dev/null
@@ -0,0 +1,128 @@
+#! /bin/bash
+# FS QA Test No. 114
+#
+# Make sure that we can handle insert-range followed by collapse-range.
+# In particular, make sure that fcollapse works for rmap when the
+# extents on either side of the collapse area are mergeable.
+#
+#-----------------------------------------------------------------------
+# Copyright (c) 2017, Oracle and/or its affiliates.  All Rights Reserved.
+#
+# 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
+# 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, write the Free Software Foundation,
+# Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+#-----------------------------------------------------------------------
+
+seq=`basename "$0"`
+seqres="$RESULT_DIR/$seq"
+echo "QA output created by $seq"
+
+here=`pwd`
+tmp=/tmp/$$
+status=1    # failure is the default!
+trap "_cleanup; exit \$status" 0 1 2 3 15
+
+_cleanup()
+{
+       cd /
+       rm -rf "$tmp".*
+}
+
+# get standard environment, filters and checks
+. ./common/rc
+. ./common/filter
+. ./common/reflink
+
+# real QA test starts here
+_supported_os Linux
+_supported_fs xfs
+_require_test_program "punch-alternating"
+_require_xfs_scratch_rmapbt
+_require_xfs_io_command "fcollapse"
+_require_xfs_io_command "finsert"
+
+rm -f "$seqres.full"
+
+echo "Format and mount"
+_scratch_mkfs > "$seqres.full" 2>&1
+_scratch_mount
+blksz=65536
+blocks1=150    # enough extents to force btree format
+blocks2=16     # local format
+len1=$((blocks1 * blksz))
+len2=$((blocks2 * blksz))
+file_blksz=$(_get_file_block_size $SCRATCH_MNT)
+
+echo "Create some files"
+$XFS_IO_PROG -f \
+       -c "falloc 0 $len1" \
+       -c "pwrite -S 0x68 -b 1048576 0 $len1" \
+       $SCRATCH_MNT/f1 >> $seqres.full
+$XFS_IO_PROG -f \
+       -c "falloc 0 $len2" \
+       -c "pwrite -S 0x68 -b 1048576 0 $len2" \
+       $SCRATCH_MNT/f2 >> $seqres.full
+
+./src/punch-alternating -o $((16 * blksz / file_blksz)) \
+       -s $((blksz / file_blksz)) \
+       -i $((blksz * 2 / file_blksz)) \
+       $SCRATCH_MNT/f1 >> "$seqres.full"
+
+echo "Insert and write file range"
+$XFS_IO_PROG \
+       -c "finsert $((8 * blksz)) $blksz" \
+       -c "pwrite -S 0x69 $((8 * blksz)) $blksz" \
+       -c "fsync" \
+       $SCRATCH_MNT/f1 >> $seqres.full
+$XFS_IO_PROG \
+       -c "finsert $((8 * blksz)) $blksz" \
+       -c "pwrite -S 0x69 $((8 * blksz)) $blksz" \
+       -c "fsync" \
+       $SCRATCH_MNT/f2 >> $seqres.full
+
+# Make reflink copies to force it to use the fancy rmap operations.
+# The test's functionality doesn't depend on the fs supporting reflink.
+_cp_reflink $SCRATCH_MNT/f1 $SCRATCH_MNT/e1 >> $seqres.full 2>&1
+_cp_reflink $SCRATCH_MNT/f2 $SCRATCH_MNT/e2 >> $seqres.full 2>&1
+
+echo "f1 bmap" >> $seqres.full
+$XFS_IO_PROG -c 'bmap -elpv' $SCRATCH_MNT/f1 >> $seqres.full
+echo "f2 bmap" >> $seqres.full
+$XFS_IO_PROG -c 'bmap -elpv' $SCRATCH_MNT/f2 >> $seqres.full
+echo "fsmap" >> $seqres.full
+$XFS_IO_PROG -c 'fsmap -v' $SCRATCH_MNT >> $seqres.full
+
+echo "Remount"
+_scratch_unmount
+_scratch_xfs_check
+_scratch_mount
+
+echo "Collapse file"
+$XFS_IO_PROG -c "fcollapse $((8 * blksz)) $blksz" $SCRATCH_MNT/f1 >> $seqres.full
+$XFS_IO_PROG -c "fcollapse $((8 * blksz)) $blksz" $SCRATCH_MNT/f2 >> $seqres.full
+
+echo "f1 bmap" >> $seqres.full
+$XFS_IO_PROG -c 'bmap -elpv' $SCRATCH_MNT/f1 >> $seqres.full
+echo "f2 bmap" >> $seqres.full
+$XFS_IO_PROG -c 'bmap -elpv' $SCRATCH_MNT/f2 >> $seqres.full
+echo "fsmap" >> $seqres.full
+$XFS_IO_PROG -c 'fsmap -v' $SCRATCH_MNT >> $seqres.full
+
+echo "Check file"
+md5sum $SCRATCH_MNT/f1 | _filter_scratch
+md5sum $SCRATCH_MNT/f2 | _filter_scratch
+od -tx1 -Ad -c $SCRATCH_MNT/f1 >> $seqres.full
+od -tx1 -Ad -c $SCRATCH_MNT/f2 >> $seqres.full
+
+# success, all done
+status=0
+exit
diff --git a/tests/xfs/114.out b/tests/xfs/114.out
new file mode 100644 (file)
index 0000000..03f73ba
--- /dev/null
@@ -0,0 +1,9 @@
+QA output created by 114
+Format and mount
+Create some files
+Insert and write file range
+Remount
+Collapse file
+Check file
+f45e899d0ed7c055f7baa4e2039eb152  SCRATCH_MNT/f1
+e57a7dd63e41d5ba02a754dd72386c3d  SCRATCH_MNT/f2
index 5977330b89aca9ec6fc89cb1d63d5c1224cef883..0a449b9a8769ff708a665cb6c5f694e0d91505dc 100644 (file)
 111 ioctl
 112 fuzzers
 113 fuzzers
+114 auto quick clone rmap
 116 quota auto quick
 117 fuzzers
 118 auto quick fsr dangerous