common/fuzzy: if the fuzz verb is random, keep fuzzing until we get a new value
[xfstests-dev.git] / common / defrag
index 732cd64006e364ac2701552d57f0e7aa70c243d4..d279382f4b56e3239a7f7d3a1e96672c9f7a521b 100644 (file)
@@ -27,7 +27,19 @@ _require_defrag()
         DEFRAG_PROG="$XFS_FSR_PROG"
        ;;
     ext4|ext4dev)
-        DEFRAG_PROG="$E4DEFRAG_PROG"
+       testfile="$TEST_DIR/$$-test.defrag"
+       donorfile="$TEST_DIR/$$-donor.defrag"
+       bsize=`_get_block_size $TEST_DIR`
+       $XFS_IO_PROG -f -c "pwrite -b $bsize 0 $bsize" $testfile > /dev/null
+       cp $testfile $donorfile
+       echo $testfile | $here/src/e4compact -v -f $donorfile | \
+               grep -q "err:95"
+       if [ $? -eq 0 ]; then
+               rm -f $testfile $donorfile 2>&1 > /dev/null
+               _notrun "$FSTYP test filesystem doesn't support online defrag"
+       fi
+       rm -f $testfile $donorfile 2>&1 > /dev/null
+       DEFRAG_PROG="$E4DEFRAG_PROG"
        ;;
     btrfs)
        DEFRAG_PROG="$BTRFS_UTIL_PROG filesystem defragment"
@@ -37,35 +49,131 @@ _require_defrag()
        ;;
     esac
 
-    _require_command $DEFRAG_PROG
+    _require_command "$DEFRAG_PROG" defragment
     _require_xfs_io_command "fiemap"
 }
 
 _extent_count()
 {
-       $XFS_IO_PROG -c "fiemap" $1 | tail -n +2 | grep -v hole | wc -l
        $XFS_IO_PROG -c "fiemap" $1  >> $seqres.full 2>&1
+       $XFS_IO_PROG -c "fiemap" $1 | tail -n +2 | grep -v hole | wc -l| $AWK_PROG '{print $1}'
+}
+
+_check_extent_count()
+{
+       min=$1
+       max=$2
+       ext_cnt=$3
+
+       # Return failure if $3 isn't between $1 and $2; let caller decide
+       # action if it's not, we just return status here.
+       if [ "$min" -gt "$ext_cnt" ]; then
+               echo "Found $ext_cnt extents  min:$max"
+               return 1;
+       fi
+       if [ "$max" -ne -1 ] && [ "$ext_cnt" -gt "$max" ]; then
+               echo "Found $ext_cnt max: $max"
+               return 1;
+       fi
+
+       if [ $max -ne $min ]; then
+           echo "in_range($min, $max)"
+       else
+           echo "$ext_cnt"
+       fi
+       # success
+       return 0
 }
 
 # Defrag file, check it, and remove it.
 _defrag()
 {
+       min_before=0
+       max_before=-1
+       min_after=0
+       max_after=-1
+       csum=1
+       mtime=1
+
+       while [ $# -gt 1 ]
+       do
+            case $1
+               in
+               --min_before)
+                   [ -z "$2" ] && _fail "missing argument for --min_before"
+                   min_before=$2
+                   shift
+                   ;;
+               --max_before)
+                   [ -z "$2" ] && _fail "missing argument for --max_before"
+                   max_before=$2
+                   shift
+                   ;;
+               --min_after)
+                   [ -z "$2" ] && _fail "missing argument for --min_after"
+                   min_after=$2
+                   shift
+                   ;;
+               --max_after)
+                   [ -z "$2" ] && _fail "missing argument for --max_after"
+                   max_after=$2
+                   shift
+                   ;;
+               --before)
+                   [ -z "$2" ] && _fail "missing argument for --before"
+                   min_before=$2
+                   max_before=$2
+                   shift
+                   ;;
+               --after)
+                   [ -z "$2" ] && _fail "missing argument for --after"
+                   min_after=$2
+                   max_after=$2
+                   shift
+                   ;;
+               --no_csum)
+                   csum=
+                   ;;
+               --no_mtime)
+                   mtime=
+                   ;;
+               --no_unlink)
+                   no_unlink=1
+                   ;;
+               *)
+                   _fail "invalid argument to common/dump function: $1"
+                   ;;
+            esac
+           shift
+       done
+
        echo -n "Before: "
-       _extent_count $1
-       CSUM_BEFORE=`md5sum $1`
+       ext_before=$(_extent_count $1)
+       _check_extent_count $min_before $max_before $ext_before
+       [ $? -eq 0 ] || _notrun "Could not adequately fragment file"
+
+       [ ! -z $csum ] && CSUM_BEFORE=`md5sum $1`
        STAT_BEFORE=`stat -c "a: %x m: %y c: %z" $1`
        $DEFRAG_PROG -v $1 >> $seqres.full 2>&1
-       _scratch_remount
+
+       _scratch_cycle_mount
        STAT_AFTER=`stat -c "a: %x m: %y c: %z" $1`
-       CSUM_AFTER=`md5sum $1`
+       [ ! -z $csum ] && CSUM_AFTER=`md5sum $1`
+
        echo -n "After: "
-       _extent_count $1
-       if [ "$CSUM_BEFORE" != "$CSUM_AFTER" ]; then
+       ext_after=$(_extent_count $1)
+       _check_extent_count $min_after $max_after $ext_after
+       [ $? -eq 0 ] || _fail "Failed to adequately defragment file"
+
+       [ "$ext_before" -lt "$ext_after" ] && \
+               _fail "Number of extents increased after defragmentation," \
+                     " before:$ext_before, after:$ext_after"
+       if [ ! -z $csum ] && [ "$CSUM_BEFORE" != "$CSUM_AFTER" ]; then
                _fail "file checksum changed post-defrag ($CSUM_BEFORE/$CSUM_AFTER)"
        fi
-       if [ "$STAT_BEFORE" != "$STAT_AFTER" ]; then
+       if [ ! -z $mtime ] && [ "$STAT_BEFORE" != "$STAT_AFTER" ]; then
                _fail "file timestamps changed post-defrag:\n$STAT_BEFORE\n$STAT_AFTER"
        fi
-       rm -f $1
+       [ -z $no_unlink ] && rm -f $1
 }