Skip to content

Commit

Permalink
xfs: register the inode cache shrinker before quotachecks
Browse files Browse the repository at this point in the history
During mount, we can do a quotacheck that involves a bulkstat pass
on all inodes. If there are more inodes in the filesystem than can
be held in memory, we require the inode cache shrinker to run to
ensure that we don't run out of memory.

Unfortunately, the inode cache shrinker is not registered until we
get to the end of the superblock setup process, which is after a
quotacheck is run if it is needed. Hence we need to register the
inode cache shrinker earlier in the mount process so that we don't
OOM during mount. This requires that we also initialise the syncd
work before we register the shrinker, so we nee dto juggle that
around as well.

While there, make sure that we have set up the block sizes in the
VFS superblock correctly before the quotacheck is run so that any
inodes that are cached as a result of the quotacheck have their
block size fields set up correctly.

Cc: stable@kernel.org
Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: Alex Elder <aelder@sgi.com>
  • Loading branch information
Dave Chinner authored and Dave Chinner committed Mar 25, 2011
1 parent 7401aaf commit 704b290
Showing 1 changed file with 24 additions and 10 deletions.
34 changes: 24 additions & 10 deletions fs/xfs/linux-2.6/xfs_super.c
Original file line number Diff line number Diff line change
Expand Up @@ -1539,17 +1539,31 @@ xfs_fs_fill_super(
if (error)
goto out_free_sb;

error = xfs_mountfs(mp);
if (error)
goto out_filestream_unmount;

/*
* we must configure the block size in the superblock before we run the
* full mount process as the mount process can lookup and cache inodes.
* For the same reason we must also initialise the syncd and register
* the inode cache shrinker so that inodes can be reclaimed during
* operations like a quotacheck that iterate all inodes in the
* filesystem.
*/
sb->s_magic = XFS_SB_MAGIC;
sb->s_blocksize = mp->m_sb.sb_blocksize;
sb->s_blocksize_bits = ffs(sb->s_blocksize) - 1;
sb->s_maxbytes = xfs_max_file_offset(sb->s_blocksize_bits);
sb->s_time_gran = 1;
set_posix_acl_flag(sb);

error = xfs_syncd_init(mp);
if (error)
goto out_filestream_unmount;

xfs_inode_shrinker_register(mp);

error = xfs_mountfs(mp);
if (error)
goto out_syncd_stop;

root = igrab(VFS_I(mp->m_rootip));
if (!root) {
error = ENOENT;
Expand All @@ -1565,14 +1579,11 @@ xfs_fs_fill_super(
goto fail_vnrele;
}

error = xfs_syncd_init(mp);
if (error)
goto fail_vnrele;

xfs_inode_shrinker_register(mp);

return 0;

out_syncd_stop:
xfs_inode_shrinker_unregister(mp);
xfs_syncd_stop(mp);
out_filestream_unmount:
xfs_filestream_unmount(mp);
out_free_sb:
Expand All @@ -1596,6 +1607,9 @@ xfs_fs_fill_super(
}

fail_unmount:
xfs_inode_shrinker_unregister(mp);
xfs_syncd_stop(mp);

/*
* Blow away any referenced inode in the filestreams cache.
* This can and will cause log traffic as inodes go inactive
Expand Down

0 comments on commit 704b290

Please sign in to comment.