#! /bin/bash # SPDX-License-Identifier: GPL-2.0 # Copyright (c) 2008 Silicon Graphics, Inc. All Rights Reserved. # # FS QA Test No. 186 # # Test out: # pv#979606: xfs bug in going from attr2 back to attr1 # # Test bug in going from attr2 back to attr1 where xfs # (due to xfs_attr_shortform_bytesfit) # would reset the di_forkoff to the m_offset instead of # leaving the di_forkoff alone as was intended. # # We create enough dirents to push us past m_attroffset, # and create an EA so we have a fork offset # and then turn on attr1 and add one more EA which # will write over the shortform dirents. # 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.* } _create_dirents() { start_num=$1 end_num=$2 cd $fork_dir for i in `seq $start_num $end_num`; do touch file.$i done } _create_eas() { start_num=$1 end_num=$2 for i in `seq $start_num $end_num`; do $SETFATTR_PROG -n user.$i -v 0xbabe $fork_dir done } _rmv_eas() { start_num=$1 end_num=$2 for i in `seq $start_num $end_num`; do $SETFATTR_PROG -x user.$i $fork_dir done } # If filetype is available (-n ftype=1) will get v3 dirs; # just filter/replace to make this look the same as if we # had v2 dirs, as we are not interested in this info. _filter_inode() { tee -a $seqres.full | \ sed -e "s/core.forkoff/forkoff/g" | \ egrep '^u.sfdir2|^u.sfdir3|^a.sfattr|forkoff' | \ egrep -v 'inumber|parent' | \ sed -e s/sfdir3/sfdir2/g | \ grep -v filetype } _filter_version() { tee -a $seqres.full | tr ',' '\n' | grep ATTR } _print_inode() { echo "" echo "=================================" _scratch_xfs_db -c "version" 2>&1 | _filter_version _scratch_xfs_db -c "inode $inum" -c p 2>&1 | _filter_inode echo "=================================" } _do_eas() { echo "" _scratch_mount if [ $1 = "-r" ]; then echo "*** remove EAs start $2 end $3 ***" _rmv_eas $2 $3 else echo "*** create EAs start $2 end $3 ***" _create_eas $2 $3 fi echo "" cd /; $UMOUNT_PROG $SCRATCH_MNT _print_inode } _do_dirents() { num=`expr $2 - $1 + 1` echo "" echo "*** create $num dirents ***" echo "" _scratch_mount _create_dirents $1 $2 cd /; $UMOUNT_PROG $SCRATCH_MNT _print_inode } _changeto_attr1() { echo "" echo "Try setting attr1 by db" echo "" _scratch_xfs_db -x -c "version attr1" | _filter_version } # get standard environment, filters and checks . ./common/rc . ./common/filter . ./common/attr # real QA test starts here # Modify as appropriate. _supported_fs xfs _supported_os Linux _require_scratch _require_attrs _require_attr_v1 rm -f $seqres.full _scratch_mkfs -i attr=2,size=512 -l lazy-count=1 | _filter_mkfs \ >>$seqres.full 2>$tmp.mkfs # import crc status and attr version . $tmp.mkfs # _scratch_mkfs may ignore $MKFS_OPTIONS if mkfs fails due to conflicts between # $MKFS_OPTIONS and the provided mkfs options, which could result in creating # an XFS we don't want. Check crc status and attr version to be sure. if [ $_fs_has_crcs -eq 1 ]; then _notrun "attr v1 not supported on $SCRATCH_DEV" fi if [ $attr -ne 2 ]; then _notrun "need attr v2 on $SCRATCH_DEV" fi # set inum to root dir ino # we'll add in dirents and EAs into the root directory inum=`_scratch_xfs_get_sb_field rootino` fork_dir=$SCRATCH_MNT _print_inode # add enough dirents to be inline but more # than will fit for m_attroffset for 512b inodes # for attr2 this is not a problem _do_dirents 1 25 # add 1 ea so we get our forkoff happening _do_eas -c 1 1 # now change back to attr1 where forkoff is constant now _changeto_attr1 # now add another EA # for a bug in xfs_add_shortform_bytesfit # where it resets the forkoff to m_attroffset>>3 instead of # leaving as di_forkoff # If it resets to m_attroffset which is in the middle of # the dirents then they will get corrupted _do_eas -c 2 2 # success, all done status=0 exit