2015-11-17 08:39:24 +11:00
|
|
|
#! /bin/bash
|
|
|
|
|
# FS QA Test No. 137
|
|
|
|
|
#
|
|
|
|
|
# Ensure that we can reflink and dedupe blocks within the same file...
|
|
|
|
|
# - Create a file with three distinct blocks ABB
|
|
|
|
|
# - Reflink block zero to the multiple-of-three blocks
|
|
|
|
|
# - Reflink block one to the multiple-of-five blocks
|
|
|
|
|
# - Dedupe block two to the multiple-of-seven blocks
|
|
|
|
|
# - Check that we successfully avoid deduping with holes, unwritten
|
|
|
|
|
# extents, and non-matches; but actually dedupe real matches.
|
|
|
|
|
#
|
|
|
|
|
#-----------------------------------------------------------------------
|
|
|
|
|
# Copyright (c) 2015, 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 /
|
2016-02-08 09:27:15 +11:00
|
|
|
rm -rf "$tmp".* "$testdir"
|
2015-11-17 08:39:24 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# get standard environment, filters and checks
|
|
|
|
|
. ./common/rc
|
|
|
|
|
. ./common/filter
|
|
|
|
|
. ./common/reflink
|
|
|
|
|
|
|
|
|
|
# real QA test starts here
|
|
|
|
|
_supported_os Linux
|
|
|
|
|
_require_test_reflink
|
|
|
|
|
_require_test_dedupe
|
|
|
|
|
_require_xfs_io_command "falloc"
|
|
|
|
|
|
|
|
|
|
rm -f "$seqres.full"
|
|
|
|
|
|
2016-02-08 09:27:15 +11:00
|
|
|
testdir="$TEST_DIR/test-$seq"
|
|
|
|
|
rm -rf "$testdir"
|
|
|
|
|
mkdir "$testdir"
|
2015-11-17 08:39:24 +11:00
|
|
|
|
|
|
|
|
echo "Create the original file blocks"
|
2016-02-08 09:27:15 +11:00
|
|
|
blksz=65536
|
|
|
|
|
_pwrite_byte 0x61 0 $blksz "$testdir/file1" >> "$seqres.full"
|
|
|
|
|
_pwrite_byte 0x62 $blksz $((blksz * 2)) "$testdir/file1" >> "$seqres.full"
|
2015-11-17 08:39:24 +11:00
|
|
|
|
2016-02-08 09:27:15 +11:00
|
|
|
nr_blks=1024
|
2015-11-17 08:39:24 +11:00
|
|
|
|
|
|
|
|
echo "fallocate half the file"
|
2016-02-08 09:27:15 +11:00
|
|
|
"$XFS_IO_PROG" -f -c "falloc $((nr_blks * blksz / 2)) $((nr_blks * blksz / 2))" "$testdir/file1" >> "$seqres.full"
|
2015-11-17 08:39:24 +11:00
|
|
|
|
|
|
|
|
echo "Reflink block zero to the threes"
|
2016-02-08 09:27:15 +11:00
|
|
|
seq 1 $((nr_blks / 3)) | while read nr; do
|
|
|
|
|
_reflink_range "$testdir/file1" 0 "$testdir/file1" $((nr * 3 * blksz)) \
|
|
|
|
|
$blksz >> "$seqres.full"
|
2015-11-17 08:39:24 +11:00
|
|
|
done
|
|
|
|
|
|
|
|
|
|
echo "Reflink block one to the fives"
|
2016-02-08 09:27:15 +11:00
|
|
|
seq 1 $((nr_blks / 5)) | while read nr; do
|
|
|
|
|
_reflink_range "$testdir/file1" $blksz "$testdir/file1" \
|
|
|
|
|
$((nr * 5 * blksz)) $blksz >> "$seqres.full"
|
2015-11-17 08:39:24 +11:00
|
|
|
done
|
|
|
|
|
|
|
|
|
|
echo "Dedupe block two to the sevens"
|
2016-02-08 09:27:15 +11:00
|
|
|
seq 1 $((nr_blks / 7)) | while read nr; do
|
|
|
|
|
_dedupe_range "$testdir/file1" $((blksz * 2)) "$testdir/file1" \
|
|
|
|
|
$((nr * 7 * blksz)) $blksz >> "$seqres.full" 2>&1
|
2015-11-17 08:39:24 +11:00
|
|
|
done
|
|
|
|
|
|
|
|
|
|
_test_remount
|
|
|
|
|
|
|
|
|
|
echo "Check block mappings"
|
2016-02-08 09:27:15 +11:00
|
|
|
md5sum "$testdir/file1" | _filter_test_dir
|
2015-11-17 08:39:24 +11:00
|
|
|
|
2016-02-08 09:27:15 +11:00
|
|
|
crcZ=$(_md5_range_checksum /dev/zero 0 $blksz)
|
|
|
|
|
crc0=$(_md5_range_checksum "$testdir/file1" 0 $blksz)
|
|
|
|
|
crc1=$(_md5_range_checksum "$testdir/file1" $blksz $blksz)
|
|
|
|
|
crc2=$(_md5_range_checksum "$testdir/file1" $((blksz * 2)) $blksz)
|
2015-11-17 08:39:24 +11:00
|
|
|
|
|
|
|
|
check_block() {
|
|
|
|
|
lblk="$1"
|
|
|
|
|
rem7=$((lblk % 7))
|
|
|
|
|
rem5=$((lblk % 5))
|
|
|
|
|
rem3=$((lblk % 3))
|
|
|
|
|
|
2016-02-08 09:27:15 +11:00
|
|
|
crc=$(_md5_range_checksum "$testdir/file1" $((lblk * blksz)) $blksz)
|
2015-11-17 08:39:24 +11:00
|
|
|
|
|
|
|
|
if [ $rem7 -eq 0 ]; then
|
|
|
|
|
if [ $rem5 -eq 0 ]; then
|
|
|
|
|
test $crc2 = $crc || echo "lblk $lblk doesn't match block 2"
|
|
|
|
|
elif [ $rem3 -eq 0 ]; then
|
|
|
|
|
test $crc0 = $crc || echo "lblk $lblk doesn't match block 0"
|
2016-02-08 09:27:15 +11:00
|
|
|
elif [ $lblk -lt $((nr_blks / 2)) ]; then
|
2015-11-17 08:39:24 +11:00
|
|
|
test $crcZ = $crc || echo "lblk $lblk isn't zeroed"
|
|
|
|
|
fi
|
|
|
|
|
elif [ $rem5 -eq 0 ]; then
|
|
|
|
|
test $crc1 = $crc || echo "lblk $lblk doesn't match block 1"
|
|
|
|
|
elif [ $rem3 -eq 0 ]; then
|
|
|
|
|
test $crc0 = $crc || echo "lblk $lblk doesn't match block 0"
|
2016-02-08 09:27:15 +11:00
|
|
|
elif [ $lblk -lt $((nr_blks / 2)) ]; then
|
2015-11-17 08:39:24 +11:00
|
|
|
test $crcZ = $crc || echo "lblk $lblk isn't zeroed"
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
2016-02-08 09:27:15 +11:00
|
|
|
seq 3 $((nr_blks - 1)) | while read lblk; do
|
2015-11-17 08:39:24 +11:00
|
|
|
err="$(check_block $lblk)"
|
|
|
|
|
test -n "$err" && echo "$lblk: $err"
|
|
|
|
|
done
|
|
|
|
|
|
|
|
|
|
# success, all done
|
|
|
|
|
status=0
|
|
|
|
|
exit
|