generic/062: don't assume same readdir order after re-creating directory
[xfstests-dev.git] / tests / generic / 062
1 #! /bin/bash
2 # FS QA Test No. 062
3 #
4 # Exercises the getfattr/setfattr tools
5 # Derived from tests originally written by Andreas Gruenbacher for ext2
6 #
7 #-----------------------------------------------------------------------
8 # Copyright (c) 2000-2002 Silicon Graphics, Inc.  All Rights Reserved.
9 #
10 # This program is free software; you can redistribute it and/or
11 # modify it under the terms of the GNU General Public License as
12 # published by the Free Software Foundation.
13 #
14 # This program is distributed in the hope that it would be useful,
15 # but WITHOUT ANY WARRANTY; without even the implied warranty of
16 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17 # GNU General Public License for more details.
18 #
19 # You should have received a copy of the GNU General Public License
20 # along with this program; if not, write the Free Software Foundation,
21 # Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
22 #
23 #-----------------------------------------------------------------------
24 #
25
26 seq=`basename $0`
27 seqres=$RESULT_DIR/$seq
28 echo "QA output created by $seq"
29
30 here=`pwd`
31 tmp=/tmp/$$
32 status=1        # failure is the default!
33
34 # get standard environment, filters and checks
35 . ./common/rc
36 . ./common/filter
37 . ./common/attr
38
39 _cleanup()
40 {
41         cd /
42         echo; echo "*** unmount"
43         _scratch_unmount 2>/dev/null
44         rm -f $tmp.*
45 }
46 trap "_cleanup; exit \$status" 0 1 2 3 15
47
48 getfattr()
49 {
50     $GETFATTR_PROG --absolute-names -dh $@ 2>&1 | _filter_scratch
51 }
52
53 setfattr()
54 {
55     $SETFATTR_PROG $@ 2>&1 | _filter_scratch
56 }
57
58 _create_test_bed()
59 {
60         echo "*** create test bed"
61         touch $SCRATCH_MNT/reg
62         mkdir -p $SCRATCH_MNT/dir
63         ln -s $SCRATCH_MNT/dir $SCRATCH_MNT/lnk
64         mkdir $SCRATCH_MNT/dev
65         mknod $SCRATCH_MNT/dev/b b 0 0
66         mknod $SCRATCH_MNT/dev/c c 1 3
67         mknod $SCRATCH_MNT/dev/p p
68         # sanity check
69         find $SCRATCH_MNT | LC_COLLATE=POSIX sort | _filter_scratch | grep -v "lost+found"
70 }
71
72 # real QA test starts here
73 _supported_fs generic
74 _supported_os Linux
75
76 _require_scratch
77 _require_attrs
78
79 rm -f $tmp.backup1 $tmp.backup2 $seqres.full
80
81 # real QA test starts here
82 _scratch_mkfs > /dev/null 2>&1 || _fail "mkfs failed"
83 _scratch_mount || _fail "mount failed"
84 _create_test_bed
85
86 # In kernels before 3.0, getxattr() fails with EPERM for an attribute which
87 # cannot exist.  Later kernels fail with ENODATA.  Accept both results.
88 invalid_attribute_filter() {
89         sed -e "s:\(No such attribute\|Operation not permitted\):No such attribute or operation not permitted:"
90 }
91
92 if [ "$USE_ATTR_SECURE" = yes ]; then
93     ATTR_MODES="user security trusted"
94 else
95     ATTR_MODES="user trusted"
96 fi
97 for nsp in $ATTR_MODES; do
98         for inode in reg dir lnk dev/b dev/c dev/p; do
99
100                 echo; echo "=== TYPE $inode; NAMESPACE $nsp"; echo
101                 echo "*** set/get one initially empty attribute"
102     
103                 setfattr -h -n $nsp.name $SCRATCH_MNT/$inode
104                 getfattr -m $nsp $SCRATCH_MNT/$inode
105
106                 echo "*** overwrite empty, set several new attributes"
107                 setfattr -h -n $nsp.name -v 0xbabe $SCRATCH_MNT/$inode
108                 setfattr -h -n $nsp.name2 -v 0xdeadbeef $SCRATCH_MNT/$inode
109                 setfattr -h -n $nsp.name3 -v 0xdeface $SCRATCH_MNT/$inode
110
111                 echo "*** fetch several attribute names and values (hex)"
112                 getfattr -m $nsp -e hex $SCRATCH_MNT/$inode
113
114                 echo "*** fetch several attribute names and values (base64)"
115                 getfattr -m $nsp -e base64 $SCRATCH_MNT/$inode
116                 
117                 echo "*** shrink value of an existing attribute"
118                 setfattr -h -n $nsp.name2 -v 0xdeaf $SCRATCH_MNT/$inode
119                 getfattr -m $nsp -e hex $SCRATCH_MNT/$inode
120
121                 echo "*** grow value of existing attribute"
122                 setfattr -h -n $nsp.name2 -v 0xdecade $SCRATCH_MNT/$inode
123                 getfattr -m $nsp -e hex $SCRATCH_MNT/$inode
124                 
125                 echo "*** set an empty value for second attribute"
126                 setfattr -h -n $nsp.name2 $SCRATCH_MNT/$inode
127                 getfattr -m $nsp -n $nsp.name2 $SCRATCH_MNT/$inode 2>&1 | invalid_attribute_filter
128
129                 echo "*** overwrite empty value"
130                 setfattr -h -n $nsp.name2 -v 0xcafe $SCRATCH_MNT/$inode
131                 getfattr -m $nsp -e hex -n $nsp.name2 $SCRATCH_MNT/$inode 2>&1 | invalid_attribute_filter
132
133                 echo "*** remove attribute"
134                 setfattr -h -x $nsp.name2 $SCRATCH_MNT/$inode
135                 getfattr -m $nsp -e hex -n $nsp.name2 $SCRATCH_MNT/$inode 2>&1 | invalid_attribute_filter
136
137                 echo "*** final list (strings, type=$inode, nsp=$nsp)"
138                 getfattr -m '.' -e hex $SCRATCH_MNT/$inode
139         
140         done
141 done
142
143
144 # Test the directory descent code
145
146 echo; echo
147
148 _extend_test_bed()
149 {
150         echo "*** extend test bed"
151         # must set some descents' attributes to be useful
152         mkdir -p $SCRATCH_MNT/here/up/ascend
153         mkdir -p $SCRATCH_MNT/descend/down/here
154         find $SCRATCH_MNT/descend | xargs setfattr -n user.x -v yz
155         find $SCRATCH_MNT/descend | xargs setfattr -n user.1 -v 23
156         find $SCRATCH_MNT/here | xargs setfattr -n trusted.a -v bc
157         find $SCRATCH_MNT/here | xargs setfattr -n trusted.9 -v 87
158         # whack a symlink in the middle, just to be difficult
159         ln -s $SCRATCH_MNT/here/up $SCRATCH_MNT/descend/and
160         # dump out our new starting point
161         find $SCRATCH_MNT | LC_COLLATE=POSIX sort | _filter_scratch | grep -v "lost+found"
162 }
163
164 _extend_test_bed
165
166 echo
167 echo "*** directory descent with us following symlinks"
168 getfattr -h -L -R -m '.' -e hex $SCRATCH_MNT | _sort_getfattr_output
169
170 echo
171 echo "*** directory descent without following symlinks"
172 getfattr -h -P -R -m '.' -e hex $SCRATCH_MNT | _sort_getfattr_output
173
174
175 # Test the backup/restore code
176
177 echo; echo
178
179 _backup()
180 {
181         # Note: we don't filter scratch here since we need to restore too.  But
182         # we *do* sort the output by path, since it otherwise would depend on
183         # readdir order, which on some filesystems may change after re-creating
184         # the files.
185         $GETFATTR_PROG --absolute-names -dh -R -m '.' $SCRATCH_MNT | _sort_getfattr_output >$1
186         echo BACKUP $1 >>$seqres.full
187         cat $1 >> $seqres.full
188         [ ! -s $1 ] && echo "warning: $1 (backup file) is empty"
189 }
190
191 echo "*** backup everything"
192 _backup $tmp.backup1
193
194 echo "*** clear out the scratch device"
195 rm -fr $SCRATCH_MNT/*
196 echo "AFTER REMOVE" >>$seqres.full
197 getfattr -L -R -m '.' $SCRATCH_MNT >>$seqres.full
198
199 echo "*** reset test bed with no extended attributes"
200 _create_test_bed
201 _extend_test_bed
202
203 echo "*** restore everything"
204 setfattr -h --restore=$tmp.backup1
205 _backup $tmp.backup2
206
207 echo "AFTER RESTORE" >>$seqres.full
208 getfattr -L -R -m '.' $SCRATCH_MNT >>$seqres.full
209
210 echo "*** compare before and after backups"
211 diff $tmp.backup1 $tmp.backup2
212 if [ $? -ne 0 ]; then
213         echo "urk, failed - creating $seq.backup1 and $seq.backup2"
214         cp $tmp.backup1 $seq.backup1 && cp $tmp.backup2 $seq.backup2
215         status=1
216         exit
217 fi
218
219 # success, all done
220 status=0
221 exit