xfs/030: hide the btree levels check errors
[xfstests-dev.git] / common / repair
1 ##/bin/bash
2 # SPDX-License-Identifier: GPL-2.0
3 # Copyright (c) 2000-2002 Silicon Graphics, Inc.  All Rights Reserved.
4 #
5 # Functions useful for xfs_repair tests
6
7 _zero_position()
8 {
9         value=$1
10         struct="$2"
11
12         # set values for off/len variables provided by db
13         eval `xfs_db -r -c "$struct" -c stack $SCRATCH_DEV | perl -ne '
14                 if (/byte offset (\d+), length (\d+)/) {
15                         print "offset=$1\nlength=$2\n"; exit
16                 }'`
17         if [ -z "$offset" -o -z "$length" ]; then
18                 echo "cannot calculate offset ($offset) or length ($length)"
19                 exit
20         fi
21         length=`expr $length / 512`
22         $here/src/devzero -v $value -b 1 -n $length -o $offset $SCRATCH_DEV \
23                 | perl -npe 's/\d\.\d\dKb/X.XXKb/g'
24 }
25
26 _filter_repair()
27 {
28         perl -ne '
29 # for sb
30 /- agno = / && next;    # remove each AG line (variable number)
31 s/(pointer to) (\d+)/\1 INO/;
32 # Changed inode output in 5.5.0
33 s/sb root inode value /sb root inode /;
34 s/realtime bitmap inode value /realtime bitmap inode /;
35 s/realtime summary inode value /realtime summary inode /;
36 s/ino pointer to /inode pointer to /;
37 #
38 s/(sb root inode) (\d+)( \(NULLFSINO\))?/\1 INO/;
39 s/(realtime bitmap inode) (\d+)( \(NULLFSINO\))?/\1 INO/;
40 s/(realtime summary inode) (\d+)( \(NULLFSINO\))?/\1 INO/;
41 s/(inconsistent with calculated value) (\d+)/\1 INO/;
42 s/\.+(found)/\1/g;      # remove "searching" output
43 # for agf + agi
44 s/(bad length -{0,1}\d+ for ag. 0, should be) (\d+)/\1 LENGTH/;
45 s/(bad length # -{0,1}\d+ for ag. 0, should be) (\d+)/\1 LENGTH/;
46 s/(bad agbno) (\d+)/\1 AGBNO/g;
47 s/(max =) (\d+)/\1 MAX/g;
48 s/(bad levels) (\d+) (for [a-z]* root, agno) (\d+)/\1 LEVELS \3 AGNO/;
49 # for root inos
50 s/(on inode) (\d+)/\1 INO/g;
51 s/(imap claims a free inode) (\d+)/\1 INO/;
52 s/(imap claims in-use inode) (\d+)/\1 INO/;
53 s/(cleared root inode) (\d+)/\1 INO/;
54 s/(resetting inode) (\d+)/\1 INO/;
55 s/(disconnected dir inode) (\d+)/\1 INO/;
56 # for log
57 s/internal log/<TYPEOF> log/g;
58 s/external log on \S+/<TYPEOF> log/g;
59 # realtime subvol - remove this whole line if it appears
60 s/        - generate realtime summary info and bitmap...\n//g;
61 #
62 # new xfs repair output filters
63 #
64 s/\s+- creating \d+ worker thread\(s\)\n//g;
65 s/\s+- reporting progress in intervals of \d+ minutes\n//g;
66 s/\s+- \d+:\d\d:\d\d:.*\n//g;
67 # 3.1.0 extra accounting output
68 /^agf_/ && next; # remove agf counts
69 /^agi_/ && next; # remove agi counts
70 /^sb_/ && next; # remove sb counts
71 /^agi unlinked/ && next; # remove agi unlinked bucket warning
72 # crc enabled filesystem output
73 /XFS_CORRUPTION_ERROR/ && next;
74 /^bad uuid/ && next;
75 /^Metadata corruption detected/ && next;
76 /^Metadata CRC error detected/ && next;
77 /^agfl has bad CRC/ && next;
78 /^bad CRC for inode/ && next;
79 # finobt enabled filesystem output
80 s/(inode chunk) (\d+)\/(\d+)/AGNO\/INO/;
81 # sunit/swidth reset messages
82 s/^(Note - .*) were copied.*/\1 fields have been reset./;
83 s/^(Please) reset (with .*) if necessary/\1 set \2/;
84 # remove new unlinked inode test
85 /^bad next_unlinked/ && next;
86 # And make them generic so we dont depend on geometry
87 s/(stripe unit) \(.*\) (and width) \(.*\)/\1 (SU) \2 (SW)/;
88 # corrupt sb messages
89 s/(superblock) (\d+)/\1 AGNO/;
90 s/(AG \#)(\d+)/\1AGNO/;
91 s/(reset bad sb for ag) (\d+)/\1 AGNO/;
92 s/(unknown block state, ag )(\d+)(, blocks? )(\d+)/\1AGNO\3AGBNO/;
93 /^Note - quota info will be regenerated on next quota mount.$/ && next;
94         print;'
95 }
96
97 # Filter out unknown block state messages that appear when rmap is enabled
98 # and we erase a btree root pointer (such that repair never finds the
99 # tree and fails to reconcile the metadata reverse mappings against the
100 # metadata).
101 _filter_repair_lostblocks() {
102         _filter_repair | sed -e '/unknown block state, ag AGNO, blocks* AGBNO/d'
103 }
104
105 _filter_dd()
106 {
107         fgrep -v records        # lose records in/out lines
108 }
109
110 # do some controlled corrupting & ensure repair recovers us
111
112 _check_repair()
113 {
114         value=$1
115         structure="$2"
116
117         #ensure the filesystem has been dirtied since last repair
118         _scratch_mount
119         POSIXLY_CORRECT=yes \
120         dd if=/bin/bash of=$SCRATCH_MNT/sh 2>&1 |_filter_dd
121         sync
122         rm -f $SCRATCH_MNT/sh
123         _scratch_unmount
124
125         _zero_position $value "$structure"
126         _scratch_xfs_repair 2>&1 | _filter_repair
127
128         # some basic sanity checks...
129         _check_scratch_fs
130         _scratch_mount                                      #mount
131         POSIXLY_CORRECT=yes \
132         dd if=/bin/bash of=$SCRATCH_MNT/sh 2>&1 |_filter_dd   #open,write
133         POSIXLY_CORRECT=yes \
134         dd if=$SCRATCH_MNT/sh of=/dev/null 2>&1 |_filter_dd #read
135         rm -f $SCRATCH_MNT/sh                               #unlink
136         _scratch_unmount                                    #umount
137 }
138
139 # make sure this script returns success
140 /bin/true