#!/usr/bin/perl # Nasty script for doing various stuff with UFS filesystems. # # Usage: # ufs.pl [-bdix] [-s ] # # By default, recursively lists inode information, blocks etc starting # at the root inode. # # -b Provide more detailed block allocation information. # -d Output file data to stdout. # -i List all inodes in numerical order, not recursively. # -s Start at the specified inode. # -x Extract each listed regular file to the # current working directory. In recursive # mode, subdirectories are created to match # the hierarchy in the filesystem. $dosub = 1; $startinode = 2; $startname = '/'; $showdata = 0; $extract = 0; $dextract = 0; $blockdetail = 0; while ($ARGV[0] =~ /^-/) { my $opt = shift; if ($opt eq "-b") { $blockdetail = 1; } elsif ($opt eq "-i") { $inodemode = 1; $dosub = 0; } elsif ($opt eq "-s") { $startinode = shift; $startname = "inode_$startinode"; } elsif ($opt eq "-d") { $showdata = 1; } elsif ($opt eq "-x") { $extract = 1; } elsif ($opt eq "-X") { $dextract = 1; } else { die "Unknown option `$opt'\n"; } } $fsdev = shift; die "No device specified\n" unless defined($fsdev); open(FS, $fsdev) || die "$fsdev: $!\n"; sub myseek { my($fh, $bn) = @_; my($ret); $how = 0; while ($bn > 2097152) { sysseek($fh, 2097152 * 512, $how); $how = 1; $bn -= 2097152; } sysseek($fh, $bn * 512, $how); return 1; } sub fsread { my($bn, $offset, $len) = @_; # print "fsread bn=$bn off=$offset len=$len\n"; if ($bn > $sb{'fs_size'} || $bn < 0) { die "fsread: eek, bad block number $bn\n"; } $bn *= ($sb{'fs_fsize'}/512); my $bn1 = int($offset/512); $bn += $bn1; $offset -= $bn1 * 512; myseek(FS, $bn) || die "myseek($bn): $!\n"; my($data); my($readlen) = ($len + 511) & ~511; # print "fsread read bn=$bn off=$offset len=$readlen\n"; die "read $len\@sector $bn: $!\n" unless (sysread(FS, $data, $readlen) == $readlen); return substr($data, $offset, $len); } sub fsread1 { my($offset, $len) = @_; my($bn) = int($offset/512); my($bo) = $offset - ($bn * 512); # print "read $len\@$offset [$bn+$bo]\n"; if (($offset & 0x1ff) || $len < 512) { myseek(FS, $bn) || die "seek($offset): $!\n"; my($data); die "readsmall $len\@$offset: $!\n" unless (sysread(FS, $data, 512) == 512); return substr($data, $offset & 0x1ff, $len); } myseek(FS, $bn) || die "seek($offset): $!\n"; my($data); die "read $len\@$offset: $!\n" unless (sysread(FS, $data, $len) == $len); return $data; } sub writeout { my ($dat, $name) = @_; my $dir; $name =~ s/^\/+//; $dir = $name; $dir =~ s/\/[^\/]+$//; if (! -d $dir) { print "making '$dir'\n"; system "mkdir -p '$dir'"; } open(OUT, "> $name") || die "$name: $!\n"; print OUT $dat; close(OUT); } sub tunpack { my($data, @spec) = @_; my($spec, $pstr, @pres, %res); for $spec (@spec) { my $s = $spec; my($cnt) = 1; $cnt = $1 if ($s =~ s/^([\d]+)//); $s =~ s/:.*//; $s = 'L2' if ($s eq 'Q'); $s = 'lL' if ($s eq 'q'); $pstr .= $s x $cnt; } @pres = unpack($pstr, $data); for $spec (@spec) { my($cnt, $s, $name) = ($spec =~ /^([\d]*)([^:]+):(.*)/); $cnt = 1 if ($cnt == 0); $cnt *= 2 if ($s eq 'Q' || $s eq 'q'); my(@res1) = splice(@pres, 0, $cnt); if ($s eq 'Q' || $s eq 'q') { my($i); for($i = 0; $i<@res1; $i++) { splice(@res1, $i, 2, $res1[$i] * 1.0 + $res1[$i + 1] * 65536.0 * 65536.0); } $cnt /= 2; } $res1[0] =~ s/\0[\0-\xff]*// if ($s =~ /^a\d+$/); if ($cnt == 1) { $res{$name} = $res1[0]; } else { $res{$name} = \@res1; } # print "$name = $res{$name}\n"; } return %res; } @sbfields = qw(l:fs_firstfield l:fs_unused_1 l:fs_sblkno l:fs_cblkno l:fs_iblkno l:fs_dblkno l:fs_old_cgoffset l:fs_old_cgmask l:fs_old_time l:fs_old_size l:fs_old_dsize l:fs_ncg l:fs_bsize l:fs_fsize l:fs_frag l:fs_minfree l:fs_old_rotdelay l:fs_old_rps l:fs_bmask l:fs_fmask l:fs_bshift l:fs_fshift l:fs_maxcontig l:fs_maxbpg l:fs_fragshift l:fs_fsbtodb l:fs_sbsize 2l:fs_spare1 l:fs_nindir l:fs_inopb l:fs_old_nspf l:fs_optim l:fs_old_npsect l:fs_old_interleave l:fs_old_trackskew l:fs_id1 l:fs_id2 l:fs_old_csaddr l:fs_cssize l:fs_cgsize l:fs_spare2 l:fs_old_nsect l:fs_old_spc l:fs_old_ncyl l:fs_old_cpg l:fs_ipg l:fs_fpg l:cs_old_ndir l:cs_old_nbfree l:cs_old_nifree l:cs_old_nffree c:fs_fmod c:fs_clean c:fs_ronly c:fs_flags a512:fs_fsmnt l:fs_cgrotor 32l:fs_csp l:fs_old_cpc l:fs_maxbsize 17q:fs_sparecon64 q:fs_sblockloc q:cs_ndir q:cs_nbfree q:cs_nifree q:cs_nffree q:cs_numclusters 3q:cs_spare q:fs_time q:fs_size q:fs_dsize q:fs_csaddr q:fs_pendingblocks l:fs_pendinginodes 20l:fs_snapinum l:fs_avgfilesize l:fs_avgfpdir l:fs_save_cgsize 27l:fs_sparecon32 l:fs_contigsumsize l:fs_maxsymlinklen l:fs_old_inodefmt Q:fs_maxfilesize q:fs_qbmask q:fs_qfmask l:fs_state l:fs_old_postblformat l:fs_old_nrpos 2l:fs_spare5 l:fs_magic); foreach (0, 8192, 65536, 262144) { $sboff = $_; undef(%sb); $sb = fsread(0, $sboff, 0x2000); %sb = tunpack($sb, @sbfields); #print "sboff $sboff magic $sb{'fs_magic'}\n"; if ($sb{'fs_magic'} == 0x011954) { $fsvers = 1; $daddrsize = 4; last; } elsif ($sb{'fs_magic'} == 0x19540119) { $fsvers = 2; $daddrsize = 8; last; } } die "No valid superblock found\n" unless defined($fsvers); print "Filesystem type is UFS$fsvers, superblock at offset $sboff\n"; print "block size $sb{'fs_bsize'} frag size $sb{'fs_fsize'}\n"; if ($fsvers == 1) { $sb{'fs_time'} = $sb{'fs_old_time'}; $sb{'fs_size'} = $sb{'fs_old_size'}; $sb{'fs_dsize'} = $sb{'fs_old_dsize'}; $ISIZE = 128; } else { $ISIZE = 256; } sub cgbase {return $sb{'fs_fpg'} * shift;} sub cgstart { my($c) = shift; return cgbase($c) + $sb{'fs_cgoffset'} * ($c & ~$sb{'fs_cgmask'}); } sub cgimin {return cgstart(shift) + $sb{'fs_iblkno'};} sub ino_to_cg {return int(shift() / $sb{'fs_ipg'});} sub blkstofrags {return shift() << $sb{'fs_fragshift'};} sub ino_to_fsba { my($x) = shift; return cgimin(ino_to_cg($x)) + blkstofrags(int(($x % $sb{'fs_ipg'}) / $sb{'fs_inopb'})) } sub ino_to_fsbo {return shift() % $sb{'fs_inopb'};} sub rdinode { my($i) = shift; if ($i < 0 || $i > $sb{'fs_ipg'} * $sb{'fs_ncg'}) { print "rdinode: bad inode $i\n"; return undef; } return fsread(ino_to_fsba($i), ino_to_fsbo($i) * $ISIZE, $ISIZE); } @ifields1 = qw(S:di_mode s:di_nlink l:lfs_inumber Q:di_size l:di_atime l:di_atimensec l:di_mtime l:di_mtimensec l:di_ctime l:di_ctimensec 12l:di_db 3l:di_ib L:di_flags l:di_blocks l:di_gen L:di_uid L:di_gid); @ifields2 = qw(S:di_mode s:di_nlink L:ldi_uid L:di_gid L:di_blksize Q:di_size Q:di_blocks q:di_gen q:di_atime q:di_mtime q:di_ctime l:di_mtimensec l:di_atimensec l:di_ctimensec L:di_kernflags L:di_flags l:di_extsize 2q:di_extb 12q:di_db 3q:di_ib 4q:di_spare); @ifields = ($fsvers == 1 ? @ifields1 : @ifields2); %ITYP = ( 4 => 'd', 8 => '-', 2 => 'c', 6 => 'b', 10 => 'l' ); sub prinode { my($inum, $nam, $dosub) = @_; my $idat = rdinode($inum); return unless defined($idat); my(%idat) = tunpack($idat, @ifields); my $typ = ($idat{'di_mode'} & 0170000) >> 12; my $mode = $idat{'di_mode'} & 07777; my $ityp = $ITYP{$typ} || '?'; my $m = $ityp; $m .= ($mode & 00400) ? 'r' : '-'; $m .= ($mode & 00200) ? 'w' : '-'; $m .= ($mode & 00100) ? ($mode & 04000) ? 's' : 'x' : ($mode & 04000) ? 'S' : '-'; $m .= ($mode & 00040) ? 'r' : '-'; $m .= ($mode & 00020) ? 'w' : '-'; $m .= ($mode & 00010) ? ($mode & 02000) ? 's' : 'x' : ($mode & 02000) ? 'S' : '-'; $m .= ($mode & 00004) ? 'r' : '-'; $m .= ($mode & 00002) ? 'w' : '-'; $m .= ($mode & 00001) ? ($mode & 01000) ? 't' : 'x' : ($mode & 01000) ? 'T' : '-'; printf("%-5d %s %3d %5d %5d %6s %s %s\n", $inum, $m, $idat{'di_nlink'}, $idat{'di_uid'}, $idat{'di_gid'}, $idat{'di_size'}, scalar(localtime($idat{'di_mtime'})), $nam); # print "idat: '$idat'\n"; printf("iaddr: frag addr %d off %d\n", ino_to_fsba($inum), ino_to_fsbo($inum) * $ISIZE); #printf("dblks: " . join(", ", @{$idat{'di_db'}}) . "\n"); #printf("iblks: " . join(", ", @{$idat{'di_ib'}}) . "\n"); if ($ityp eq 'l' && $idat{'di_size'} < (15*4)) { printf("-> '%s'\n", substr($idat, 40, $idat{'di_size'})); next; } my $dat; my(@blocks) = iblocks(\%idat); print "blocks: " . join(" ", @blocks) . "\n"; if ($showdata || $extract || $ityp eq 'd') { my($b, $bn, $lbn); foreach $b (@blocks) { ($lbn, $bn) = split(":", $b); next if ($bn =~ /^BAD/ || $lbn =~ /^INDIR/); # print "read lbn $lbn block $bn\n"; substr($dat, $lbn * $sb{'fs_bsize'}, $sb{'fs_bsize'}) = fsread($bn, 0, $sb{'fs_bsize'}); } } $dat = substr($dat, 0, $idat{'di_size'}); if ($ITYP{$typ} eq 'd') { prdir($dat, $nam, $dosub); writeout($dat, "dirinode_$inum") if ($dextract); } elsif ($ITYP{$typ} eq '-') { print "file contents: '$dat'\n" if ($showdata); writeout($dat, $nam) if ($extract); } } sub blocks { my($bn, $lbn, $level) = @_; my(@blocks); if ($level == 0) { return ("$lbn:BAD;block=$bn") if ($bn <= 0 || $bn > $sb{'fs_size'}); return ("$lbn:$bn"); } return ("INDIR$level;$lbn:BAD;block=$bn") if ($bn <= 0 || $bn > $sb{'fs_size'}); push(@blocks, "INDIR$level;$lbn:$bn"); my $bdat = fsread($bn, 0, $sb{'fs_bsize'}); my $bfmt = $sb{'fs_nindir'} . ($fsvers == 1 ? "l" : "q"); #print "bfmt: $bfmt\n"; my %bn = tunpack($bdat, "$bfmt:indir"); for $bn (@{$bn{'indir'}}) { push(@blocks, blocks($bn, $lbn, $level - 1)) if ($bn != 0); $lbn += ($sb{'fs_bsize'} / $daddrsize) ** ($level - 1); } return @blocks; } sub iblocks { my($idat) = @_; my($i, $n); my(@blocks); my $lbn = 0; for ($i = 0; $i < 12; $i++) { my $bn = $idat->{'di_db'}->[$i]; push(@blocks, blocks($bn, $lbn, 0)) if ($bn != 0); $lbn++; } for ($i = 0; $i < 3; $i++) { my $bn = $idat->{'di_ib'}->[$i]; push(@blocks, blocks($bn, $lbn, $i + 1)) if ($bn != 0); $lbn += ($sb{'fs_bsize'} / $daddrsize) ** ($i + 1); } return @blocks; } sub prdir { my($dat, $nam, $dosub) = @_; my(%dir); my($off); $off = 0; while (length($dat) > 0) { ($dir{'d_ino'}, $dir{'d_reclen'}, $dir{'d_type'}, $dir{'d_namlen'}) = unpack("LSCC", $dat); $dir{'d_name'} = substr($dat, 8, $dir{'d_namlen'}); if ($dir{'d_reclen'} == 0 || $dir{'d_reclen'} > (($off + 512) & ~511) - $pos) { print "bad entry at offset $off\n"; $dir{'d_reclen'} = ($off + 512) & ~511; $off += $dir{'d_reclen'}; $dat = substr($dat, $dir{'d_reclen'}); next; } $off += $dir{'d_reclen'}; $dat = substr($dat, $dir{'d_reclen'}); next if ($dir{'d_ino'} == 0); printf("ino %d reclen 0x%x type 0%o namelen %d name '%s'\n", $dir{'d_ino'}, $dir{'d_reclen'}, $dir{'d_type'}, $dir{'d_namlen'}, $dir{'d_name'}); if ($dir{'d_name'} ne '.' && $dir{'d_name'} ne '..' && $dosub) { prinode($dir{'d_ino'}, ($nam eq "/" ? "" : $nam)."/".$dir{'d_name'}, $dosub); } } print "----- end of $nam -----\n"; } if ($inodemode) { for ($i = $startinode; $i < $sb{'fs_ipg'} * $sb{'fs_ncg'}; $i++) { my $offset = ino_to_fsba($i) * $sb{'fs_fsize'} + ino_to_fsbo($i) * $ISIZE; my($bn) = int($offset/512); my($bo) = $offset - ($bn * 512); print "inode $i: $offset [$bn+$bo]\n"; prinode($i, "inode_$i", $dosub); } } else { prinode($startinode, $startname, $dosub); }