[PATCH] bfs iget() abuses
bfs_fill_super() walks the inode table to get the bitmap of free inodes and collect stats. It has no business using iget() for that - it's a lot of extra work, extra icache pollution and more complex code. Switched to walking the damn thing directly. Note: that also allows to kill ->i_dsk_ino in there - separate patch if Tigran can confirm that this field can be zero only for deleted inodes (i.e. something that could only be found during that scan and not by normal lookups). Signed-off-by: Al Viro <viro@zeniv.linux.org.uk> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
This commit is contained in:
parent
ce0fe7e70a
commit
c2b513dfbb
1 changed files with 31 additions and 13 deletions
|
@ -362,23 +362,41 @@ static int bfs_fill_super(struct super_block *s, void *data, int silent)
|
||||||
info->si_lf_eblk = 0;
|
info->si_lf_eblk = 0;
|
||||||
info->si_lf_sblk = 0;
|
info->si_lf_sblk = 0;
|
||||||
info->si_lf_ioff = 0;
|
info->si_lf_ioff = 0;
|
||||||
|
bh = NULL;
|
||||||
for (i=BFS_ROOT_INO; i<=info->si_lasti; i++) {
|
for (i=BFS_ROOT_INO; i<=info->si_lasti; i++) {
|
||||||
inode = iget(s,i);
|
struct bfs_inode *di;
|
||||||
if (BFS_I(inode)->i_dsk_ino == 0)
|
int block = (i - BFS_ROOT_INO)/BFS_INODES_PER_BLOCK + 1;
|
||||||
|
int off = (i - BFS_ROOT_INO) % BFS_INODES_PER_BLOCK;
|
||||||
|
unsigned long sblock, eblock;
|
||||||
|
|
||||||
|
if (!off) {
|
||||||
|
brelse(bh);
|
||||||
|
bh = sb_bread(s, block);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!bh)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
di = (struct bfs_inode *)bh->b_data + off;
|
||||||
|
|
||||||
|
if (!di->i_ino) {
|
||||||
info->si_freei++;
|
info->si_freei++;
|
||||||
else {
|
continue;
|
||||||
|
}
|
||||||
set_bit(i, info->si_imap);
|
set_bit(i, info->si_imap);
|
||||||
info->si_freeb -= inode->i_blocks;
|
info->si_freeb -= BFS_FILEBLOCKS(di);
|
||||||
if (BFS_I(inode)->i_eblock > info->si_lf_eblk) {
|
|
||||||
info->si_lf_eblk = BFS_I(inode)->i_eblock;
|
sblock = le32_to_cpu(di->i_sblock);
|
||||||
info->si_lf_sblk = BFS_I(inode)->i_sblock;
|
eblock = le32_to_cpu(di->i_eblock);
|
||||||
|
if (eblock > info->si_lf_eblk) {
|
||||||
|
info->si_lf_eblk = eblock;
|
||||||
|
info->si_lf_sblk = sblock;
|
||||||
info->si_lf_ioff = BFS_INO2OFF(i);
|
info->si_lf_ioff = BFS_INO2OFF(i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
iput(inode);
|
brelse(bh);
|
||||||
}
|
|
||||||
if (!(s->s_flags & MS_RDONLY)) {
|
if (!(s->s_flags & MS_RDONLY)) {
|
||||||
mark_buffer_dirty(bh);
|
mark_buffer_dirty(info->si_sbh);
|
||||||
s->s_dirt = 1;
|
s->s_dirt = 1;
|
||||||
}
|
}
|
||||||
dump_imap("read_super", s);
|
dump_imap("read_super", s);
|
||||||
|
|
Loading…
Reference in a new issue