xfs: regression tests for reflink quota bugs

Add three tests to look for quota bugs in xfs reflink. The first
test looks for problems when we have speculative cow reservations in
memory, we chown the file, but the reservations don't move to the
new owner.  The second test checks that we remembered to dqattach
the inodes before performing reflink operations. The third is a
stress test for reflink quota handling near enospc and helped us to
find a directio cow write corruption bug when free space is
fragmented.

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>
This commit is contained in:
Darrick J. Wong
2018-02-07 13:19:38 -08:00
committed by Eryu Guan
parent 60ba1274f5
commit f2d7e7573b
7 changed files with 339 additions and 0 deletions
Executable
+93
View File
@@ -0,0 +1,93 @@
#! /bin/bash
# FS QA Test No. 440
#
# Regression test for a quota accounting bug when changing the owner of
# a file that has CoW reservations and no dirty pages. The reservations
# should shift over to the new owner, but they do not.
#
#-----------------------------------------------------------------------
# Copyright (c) 2018 Oracle, Inc. 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 -f $tmp.*
}
# get standard environment, filters and checks
. ./common/rc
. ./common/reflink
. ./common/quota
. ./common/filter
# Modify as appropriate.
_supported_fs xfs
_supported_os Linux
_require_quota
_require_scratch_reflink
_require_cp_reflink
_require_user
rm -f $seqres.full
echo "Format and mount"
_scratch_mkfs > "$seqres.full" 2>&1
_scratch_mount "-o usrquota,grpquota" >> "$seqres.full" 2>&1
echo "Create files"
$XFS_IO_PROG -c "cowextsize 1m" $SCRATCH_MNT
touch $SCRATCH_MNT/a $SCRATCH_MNT/force_fsgqa
chown $qa_user $SCRATCH_MNT/a $SCRATCH_MNT/force_fsgqa
_pwrite_byte 0x58 0 64k $SCRATCH_MNT/a >> $seqres.full
$XFS_IO_PROG -c 'stat -r' $SCRATCH_MNT/a | grep stat.size >> $seqres.full
_report_quota_blocks "-u $SCRATCH_MNT"
echo "Reflink and CoW"
_cp_reflink $SCRATCH_MNT/a $SCRATCH_MNT/b
_pwrite_byte 0x59 0 64k $SCRATCH_MNT/a >> $seqres.full
$XFS_IO_PROG -c 'stat -r' $SCRATCH_MNT/a | grep stat.size >> $seqres.full
_report_quota_blocks "-u $SCRATCH_MNT"
echo "Sync"
sync
_report_quota_blocks "-u $SCRATCH_MNT"
echo "Chown and check quota"
chown root $SCRATCH_MNT/a
$XFS_IO_PROG -c 'stat -r' $SCRATCH_MNT/a | grep stat.size >> $seqres.full
_report_quota_blocks "-u $SCRATCH_MNT"
echo "Remount"
_scratch_unmount
_scratch_mount "-o usrquota,grpquota" >> "$seqres.full" 2>&1
$XFS_IO_PROG -c 'stat -r' $SCRATCH_MNT/a | grep stat.size >> $seqres.full
_report_quota_blocks "-u $SCRATCH_MNT"
# success, all done
status=0
exit
+17
View File
@@ -0,0 +1,17 @@
QA output created by 440
Format and mount
Create files
root 0 0 0
fsgqa 64 0 0
Reflink and CoW
root 0 0 0
fsgqa 1152 0 0
Sync
root 0 0 0
fsgqa 1088 0 0
Chown and check quota
root 1024 0 0
fsgqa 64 0 0
Remount
root 64 0 0
fsgqa 64 0 0
Executable
+88
View File
@@ -0,0 +1,88 @@
#! /bin/bash
# FS QA Test No. 441
#
# Regression test for a quota accounting bug when reflinking across EOF
# of a file in which we forgot dq_attach.
#
#-----------------------------------------------------------------------
# Copyright (c) 2018 Oracle, Inc. 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 -f $tmp.*
}
# get standard environment, filters and checks
. ./common/rc
. ./common/reflink
. ./common/quota
. ./common/filter
# Modify as appropriate.
_supported_fs xfs
_supported_os Linux
_require_quota
_require_scratch_reflink
_require_cp_reflink
_require_user
rm -f $seqres.full
check_quota() {
du_total="$(du -ksc $SCRATCH_MNT/a $SCRATCH_MNT/b | tail -n 1 | awk '{print $1}')"
qu_total="$(_report_quota_blocks "-u $SCRATCH_MNT" | grep $qa_user | awk '{print $2}')"
echo "du: $du_total; quota: $qu_total"
}
echo "Format and mount (noquota)"
_scratch_mkfs > "$seqres.full" 2>&1
_scratch_mount "-o noquota" >> $seqres.full 2>&1
echo "Create files"
_pwrite_byte 0x58 0 1m $SCRATCH_MNT/a >> $seqres.full
_pwrite_byte 0x58 0 1m $SCRATCH_MNT/b >> $seqres.full
chown $qa_user $SCRATCH_MNT/a $SCRATCH_MNT/b
check_quota 2>&1 | _filter_scratch
echo "Mount (usrquota) and check quota"
_scratch_unmount
_scratch_mount "-o usrquota,grpquota" >> "$seqres.full" 2>&1
check_quota
echo "Reflink and check quota again"
_reflink_range $SCRATCH_MNT/a 256k $SCRATCH_MNT/b 384k 128k >> $seqres.full
_reflink_range $SCRATCH_MNT/a 256k $SCRATCH_MNT/b 960k 128k >> $seqres.full
check_quota
echo "Force quotacheck"
_check_quota_usage
# success, all done
status=0
exit
+13
View File
@@ -0,0 +1,13 @@
QA output created by 441
Format and mount (noquota)
Create files
repquota: Mountpoint (or device) SCRATCH_MNT not found or has no quota enabled.
repquota: Not all specified mountpoints are using quota.
du: 2048; quota:
Mount (usrquota) and check quota
du: 2048; quota: 2048
Reflink and check quota again
du: 2112; quota: 2112
Force quotacheck
Comparing user usage
Comparing group usage
Executable
+119
View File
@@ -0,0 +1,119 @@
#! /bin/bash
# FS QA Test No. 442
#
# Force enable all XFS quotas, run fsstress until the fs runs out of
# space, and make sure the quotas are still correct when we're done.
# This is a general regression/stress test for numerous quota bugs with
# reflink and copy on write.
#
#-----------------------------------------------------------------------
# Copyright (c) 2018 Oracle, Inc. 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 -f $tmp.*
$KILLALL_PROG -9 fsstress > /dev/null 2>&1
}
# get standard environment, filters and checks
. ./common/rc
. ./common/quota
. ./common/filter
. ./common/reflink
# Modify as appropriate.
_supported_fs xfs
_supported_os Linux
_require_scratch_reflink
_require_quota
_require_command "$KILLALL_PROG" "killall"
rm -f $seqres.full
report_quota_blocks() {
$XFS_QUOTA_PROG -x -c "report $1" $SCRATCH_MNT | \
awk '{x += $2;} END { print(x); }'
}
compare_quota_to_du() {
test $1 -eq $2 || echo "$3 quota $2 blocks does not match du $1 blocks?"
}
# Make sure the user/group/project quota block counts match the du output.
# This ensures that we did the quota accounting correctly and that we're
# accurately reporting cow preallocation blocks in stat.
check_quota_du_blocks() {
sync
#$XFS_QUOTA_PROG -x -c 'report' $SCRATCH_MNT >> $seqres.full
du_rep=$(du -ks $SCRATCH_MNT | awk '{print $1}')
u_rep=$(report_quota_blocks -u)
g_rep=$(report_quota_blocks -g)
p_rep=$(report_quota_blocks -p)
compare_quota_to_du $du_rep $u_rep "user"
compare_quota_to_du $du_rep $g_rep "group"
compare_quota_to_du $du_rep $p_rep "project"
}
echo "Format and fsstress"
_qmount_option "usrquota,grpquota,prjquota"
# We use a small volume so that we hit ENOSPC. This is critical for
# regression testing a bug in the directio write code that could result in fs
# corruption ("xfs: check reflink allocation mappings").
#
# This started as a test for quota accounting problems ("xfs: treat CoW fork
# operations as delalloc for quota accounting") and ("xfs: call
# xfs_qm_dqattach before performing reflink operations") though each of those
# tests now have separate faster-running regression tests.
_scratch_mkfs_sized $((1600 * 1048576)) > $seqres.full 2>&1
_scratch_mount >> $seqres.full 2>&1 || _fail "mount failed..."
nr_cpus=$((LOAD_FACTOR * 4))
nr_ops=$((25000 * nr_cpus * TIME_FACTOR))
$FSSTRESS_PROG $FSSTRESS_AVOID -w -d $SCRATCH_MNT -n $nr_ops -p $nr_cpus >> $seqres.full
echo "Check quota before remount"
check_quota_du_blocks
# Clear out all the preallocations before we quotacheck.
# The count comparison in _check_quota_usage will be unhappy if we don't
# manage to clean out all the cow preallocations before the remount.
_scratch_unmount
_scratch_mount
# Make sure the usage doesn't change after quotacheck.
echo "Check quota after remount"
_check_quota_usage
check_quota_du_blocks
# success, all done
status=0
exit
+6
View File
@@ -0,0 +1,6 @@
QA output created by 442
Format and fsstress
Check quota before remount
Check quota after remount
Comparing user usage
Comparing group usage
+3
View File
@@ -437,3 +437,6 @@
437 auto quick other
438 auto quick quota dangerous
439 auto quick fuzzers log
440 auto quick clone quota
441 auto quick clone quota
442 auto stress clone quota