blob: e0992055959fe758129ad84be59bb581676f1dd3 [file] [log] [blame]
#!/usr/bin/perl -w
# SPDX-License-Identifier: GPL-2.0
# Copyright (c) 2000-2001 Silicon Graphics, Inc. All Rights Reserved.
#
# use db to try to traverse the entire filesystem starting at the root
#
my $device;
my $rootino;
my $agcount;
my $versionnum;
my $dir_version;
my @dir_inodes;
my @bmap_blocks;
my @block_inodes;
my $mode;
sub db($)
{
my ($args)=@_;
my ($ret);
$ret=`xfs_db -r $args $device 2> /dev/null`;
die "ERROR executing xfs_db -r $args $device" if ($?);
return $ret;
}
sub fmt($)
{
my ($text)=@_;
my $c=0;
print " ";
foreach (split("\n",$text)) {
s/^core\.//;
if ($c+length($_) >= 70) {
$c=0;
print ",\n ";
}
if ($c) {
print ", ";
$c+=2;
}
print "$_";
$c+=length($_)+2;
}
print "\n";
}
sub inode($)
{
my ($num)=@_;
my ($t);
@dir_inodes=();
$t=db("-c \"inode $num\" -c \"print\"");
print " *** Inode $num\n";
fmt($t);
($mode)= $t=~ /^core.mode = (\d+)$/m;
if ($t=~ /a\.bmx/m) {
bmap("inode $num","attr");
foreach (@bmap_blocks) {
attr_block($_);
}
}
if (eval "$mode & 040000") {
if ( $t=~ /sfdir/m) {
while ($t=~ /inumber(?:\.i[48])? = (\d+)$/mg) {
push(@dir_inodes,$1);
}
}
if ( $t=~ /u\.bmx/m) {
bmap("inode $num","dir");
foreach (@bmap_blocks) {
dir_block($_);
push(@dir_inodes,@block_inodes);
}
}
} else {
bmap("inode $num","file") if ( $t=~ /u\.bmx/m);
}
}
sub bmap($$)
{
my ($cmd,$type)=@_;
my ($t);
@bmap_blocks=();
$flag=($type eq "attr")?"-a":"";
$t=db("-c \"$cmd\" -c \"bmap $flag\"");
print " *** bmap $type $cmd\n";
fmt($t);
if ($type eq "dir" || $type eq "attr") {
while ($t=~ /startblock (\d+) \(.+\) count (\d+)/mg) {
for ($b=$1;$b<$1+$2;$b++) {
push(@bmap_blocks,$b);
}
}
}
}
sub dir_block($)
{
my ($num)=@_;
my ($t);
@block_inodes=();
$type=($dir_version==2)?"dir2":"dir";
$t=db("-c \"fsblock $num\" -c \"type $type\" -c \"print\"");
print " *** $type block $num\n";
# need to drop . and ..
($self)= $t=~ /\[(\d+)\].name = \"\.\"/m;
($parent)= $t=~ /\[(\d+)\].name = \"\.\.\"/m;
fmt($t);
while ($t=~ /\[(\d+)\].inumber = (\d+)/mg) {
next if (defined $self && $1 == $self);
next if (defined $parent && $1 == $parent);
push(@block_inodes, $2);
}
}
sub attr_block($)
{
my ($num)=@_;
my ($t);
$t=db("-c \"fsblock $num\" -c \"type attr\" -c \"print\"");
print " *** attr block $num\n";
fmt($t);
}
sub sb($)
{
my ($num)=@_;
my ($t);
$t=db("-c \"sb $num\" -c \"print\"");
print " *** SB $num\n";
fmt($t);
($rootino)= $t=~ /^rootino = (\d+)$/m;
($agcount)= $t=~ /^agcount = (\d+)$/m;
($versionnum)= $t=~ /^versionnum = (0x[\da-f]+)$/m;
$dir_version = (eval "$versionnum & 0x2000")?2:1;
}
die "Usage: $0 <XFS device>\n" unless (@ARGV == 1);
$device=shift @ARGV;
die "can't read $device\n" unless (-r $device);
die "$device is not a block device\n" unless (-b _);
chomp($HOST = `hostname -s`);
print "*** db-walk host $HOST device $device\n";
sb(0);
for ($ag=1;$ag<$agcount;$ag++) {
sb($ag);
}
@inodes=($rootino);
while ($_ = shift @inodes) {
inode($_);
push(@inodes,@dir_inodes);
}