mirror of
https://github.com/followmsi/android_kernel_google_msm.git
synced 2024-11-06 23:17:41 +00:00
f2fs: don't grab super block buffer header all the time
We have already got one copy of valid super block in memory, do not grab buffer header of super block all the time. Signed-off-by: Chao Yu <chao2.yu@samsung.com> Signed-off-by: Jaegeuk Kim <jaegeuk@kernel.org>
This commit is contained in:
parent
ba9a1899aa
commit
55ead4163c
3 changed files with 39 additions and 54 deletions
|
@ -717,8 +717,8 @@ enum {
|
||||||
struct f2fs_sb_info {
|
struct f2fs_sb_info {
|
||||||
struct super_block *sb; /* pointer to VFS super block */
|
struct super_block *sb; /* pointer to VFS super block */
|
||||||
struct proc_dir_entry *s_proc; /* proc entry */
|
struct proc_dir_entry *s_proc; /* proc entry */
|
||||||
struct buffer_head *raw_super_buf; /* buffer head of raw sb */
|
|
||||||
struct f2fs_super_block *raw_super; /* raw super block pointer */
|
struct f2fs_super_block *raw_super; /* raw super block pointer */
|
||||||
|
int valid_super_block; /* valid super block no */
|
||||||
int s_flag; /* flags for sbi */
|
int s_flag; /* flags for sbi */
|
||||||
|
|
||||||
/* for node-related operations */
|
/* for node-related operations */
|
||||||
|
|
|
@ -1611,20 +1611,16 @@ static int f2fs_ioc_get_encryption_pwsalt(struct file *filp, unsigned long arg)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
/* update superblock with uuid */
|
/* update superblock with uuid */
|
||||||
lock_buffer(sbi->raw_super_buf);
|
|
||||||
generate_random_uuid(sbi->raw_super->encrypt_pw_salt);
|
generate_random_uuid(sbi->raw_super->encrypt_pw_salt);
|
||||||
unlock_buffer(sbi->raw_super_buf);
|
|
||||||
|
|
||||||
err = f2fs_commit_super(sbi, false);
|
err = f2fs_commit_super(sbi, false);
|
||||||
|
|
||||||
mnt_drop_write_file(filp);
|
|
||||||
if (err) {
|
if (err) {
|
||||||
/* undo new data */
|
/* undo new data */
|
||||||
lock_buffer(sbi->raw_super_buf);
|
|
||||||
memset(sbi->raw_super->encrypt_pw_salt, 0, 16);
|
memset(sbi->raw_super->encrypt_pw_salt, 0, 16);
|
||||||
unlock_buffer(sbi->raw_super_buf);
|
mnt_drop_write_file(filp);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
mnt_drop_write_file(filp);
|
||||||
got_it:
|
got_it:
|
||||||
if (copy_to_user((__u8 __user *)arg, sbi->raw_super->encrypt_pw_salt,
|
if (copy_to_user((__u8 __user *)arg, sbi->raw_super->encrypt_pw_salt,
|
||||||
16))
|
16))
|
||||||
|
|
|
@ -560,7 +560,6 @@ static void f2fs_put_super(struct super_block *sb)
|
||||||
wait_for_completion(&sbi->s_kobj_unregister);
|
wait_for_completion(&sbi->s_kobj_unregister);
|
||||||
|
|
||||||
sb->s_fs_info = NULL;
|
sb->s_fs_info = NULL;
|
||||||
brelse(sbi->raw_super_buf);
|
|
||||||
kfree(sbi->raw_super);
|
kfree(sbi->raw_super);
|
||||||
kfree(sbi);
|
kfree(sbi);
|
||||||
}
|
}
|
||||||
|
@ -1125,65 +1124,53 @@ static void init_sb_info(struct f2fs_sb_info *sbi)
|
||||||
*/
|
*/
|
||||||
static int read_raw_super_block(struct super_block *sb,
|
static int read_raw_super_block(struct super_block *sb,
|
||||||
struct f2fs_super_block **raw_super,
|
struct f2fs_super_block **raw_super,
|
||||||
struct buffer_head **raw_super_buf,
|
int *valid_super_block, int *recovery)
|
||||||
int *recovery)
|
|
||||||
{
|
{
|
||||||
int block = 0;
|
int block = 0;
|
||||||
struct buffer_head *buffer;
|
struct buffer_head *bh;
|
||||||
struct f2fs_super_block *super;
|
struct f2fs_super_block *super, *buf;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
super = kzalloc(sizeof(struct f2fs_super_block), GFP_KERNEL);
|
super = kzalloc(sizeof(struct f2fs_super_block), GFP_KERNEL);
|
||||||
if (!super)
|
if (!super)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
retry:
|
retry:
|
||||||
buffer = sb_bread(sb, block);
|
bh = sb_bread(sb, block);
|
||||||
if (!buffer) {
|
if (!bh) {
|
||||||
*recovery = 1;
|
*recovery = 1;
|
||||||
f2fs_msg(sb, KERN_ERR, "Unable to read %dth superblock",
|
f2fs_msg(sb, KERN_ERR, "Unable to read %dth superblock",
|
||||||
block + 1);
|
block + 1);
|
||||||
if (block == 0) {
|
err = -EIO;
|
||||||
block++;
|
goto next;
|
||||||
goto retry;
|
|
||||||
} else {
|
|
||||||
err = -EIO;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy(super, buffer->b_data + F2FS_SUPER_OFFSET, sizeof(*super));
|
buf = (struct f2fs_super_block *)(bh->b_data + F2FS_SUPER_OFFSET);
|
||||||
|
|
||||||
/* sanity checking of raw super */
|
/* sanity checking of raw super */
|
||||||
if (sanity_check_raw_super(sb, super)) {
|
if (sanity_check_raw_super(sb, buf)) {
|
||||||
brelse(buffer);
|
brelse(bh);
|
||||||
*recovery = 1;
|
*recovery = 1;
|
||||||
f2fs_msg(sb, KERN_ERR,
|
f2fs_msg(sb, KERN_ERR,
|
||||||
"Can't find valid F2FS filesystem in %dth superblock",
|
"Can't find valid F2FS filesystem in %dth superblock",
|
||||||
block + 1);
|
block + 1);
|
||||||
if (block == 0) {
|
err = -EINVAL;
|
||||||
block++;
|
goto next;
|
||||||
goto retry;
|
|
||||||
} else {
|
|
||||||
err = -EINVAL;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!*raw_super) {
|
if (!*raw_super) {
|
||||||
*raw_super_buf = buffer;
|
memcpy(super, buf, sizeof(*super));
|
||||||
|
*valid_super_block = block;
|
||||||
*raw_super = super;
|
*raw_super = super;
|
||||||
} else {
|
|
||||||
/* already have a valid superblock */
|
|
||||||
brelse(buffer);
|
|
||||||
}
|
}
|
||||||
|
brelse(bh);
|
||||||
|
|
||||||
|
next:
|
||||||
/* check the validity of the second superblock */
|
/* check the validity of the second superblock */
|
||||||
if (block == 0) {
|
if (block == 0) {
|
||||||
block++;
|
block++;
|
||||||
goto retry;
|
goto retry;
|
||||||
}
|
}
|
||||||
|
|
||||||
out:
|
|
||||||
/* No valid superblock */
|
/* No valid superblock */
|
||||||
if (!*raw_super) {
|
if (!*raw_super) {
|
||||||
kfree(super);
|
kfree(super);
|
||||||
|
@ -1196,18 +1183,16 @@ out:
|
||||||
int f2fs_commit_super(struct f2fs_sb_info *sbi, bool recover)
|
int f2fs_commit_super(struct f2fs_sb_info *sbi, bool recover)
|
||||||
{
|
{
|
||||||
struct f2fs_super_block *super = F2FS_RAW_SUPER(sbi);
|
struct f2fs_super_block *super = F2FS_RAW_SUPER(sbi);
|
||||||
struct buffer_head *sbh = sbi->raw_super_buf;
|
|
||||||
struct buffer_head *bh;
|
struct buffer_head *bh;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
/* write back-up superblock first */
|
/* write back-up superblock first */
|
||||||
bh = sb_getblk(sbi->sb, sbh->b_blocknr ? 0 : 1);
|
bh = sb_getblk(sbi->sb, sbi->valid_super_block ? 0 : 1);
|
||||||
if (!bh)
|
if (!bh)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
lock_buffer(bh);
|
lock_buffer(bh);
|
||||||
memcpy(bh->b_data + F2FS_SUPER_OFFSET, super, sizeof(*super));
|
memcpy(bh->b_data + F2FS_SUPER_OFFSET, super, sizeof(*super));
|
||||||
WARN_ON(sbh->b_size != F2FS_BLKSIZE);
|
|
||||||
set_buffer_uptodate(bh);
|
set_buffer_uptodate(bh);
|
||||||
set_buffer_dirty(bh);
|
set_buffer_dirty(bh);
|
||||||
unlock_buffer(bh);
|
unlock_buffer(bh);
|
||||||
|
@ -1220,33 +1205,37 @@ int f2fs_commit_super(struct f2fs_sb_info *sbi, bool recover)
|
||||||
if (recover || err)
|
if (recover || err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
/* write current valid superblock */
|
bh = sb_getblk(sbi->sb, sbi->valid_super_block);
|
||||||
lock_buffer(sbh);
|
if (!bh)
|
||||||
if (memcmp(sbh->b_data + F2FS_SUPER_OFFSET, super, sizeof(*super))) {
|
return -EIO;
|
||||||
f2fs_msg(sbi->sb, KERN_INFO, "Write modified valid superblock");
|
|
||||||
memcpy(sbh->b_data + F2FS_SUPER_OFFSET, super, sizeof(*super));
|
|
||||||
}
|
|
||||||
set_buffer_dirty(sbh);
|
|
||||||
unlock_buffer(sbh);
|
|
||||||
|
|
||||||
return __sync_dirty_buffer(sbh, WRITE_FLUSH_FUA);
|
/* write current valid superblock */
|
||||||
|
lock_buffer(bh);
|
||||||
|
memcpy(bh->b_data + F2FS_SUPER_OFFSET, super, sizeof(*super));
|
||||||
|
set_buffer_uptodate(bh);
|
||||||
|
set_buffer_dirty(bh);
|
||||||
|
unlock_buffer(bh);
|
||||||
|
|
||||||
|
err = __sync_dirty_buffer(bh, WRITE_FLUSH_FUA);
|
||||||
|
brelse(bh);
|
||||||
|
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int f2fs_fill_super(struct super_block *sb, void *data, int silent)
|
static int f2fs_fill_super(struct super_block *sb, void *data, int silent)
|
||||||
{
|
{
|
||||||
struct f2fs_sb_info *sbi;
|
struct f2fs_sb_info *sbi;
|
||||||
struct f2fs_super_block *raw_super;
|
struct f2fs_super_block *raw_super;
|
||||||
struct buffer_head *raw_super_buf;
|
|
||||||
struct inode *root;
|
struct inode *root;
|
||||||
long err;
|
long err;
|
||||||
bool retry = true, need_fsck = false;
|
bool retry = true, need_fsck = false;
|
||||||
char *options = NULL;
|
char *options = NULL;
|
||||||
int recovery, i;
|
int recovery, i, valid_super_block;
|
||||||
|
|
||||||
try_onemore:
|
try_onemore:
|
||||||
err = -EINVAL;
|
err = -EINVAL;
|
||||||
raw_super = NULL;
|
raw_super = NULL;
|
||||||
raw_super_buf = NULL;
|
valid_super_block = -1;
|
||||||
recovery = 0;
|
recovery = 0;
|
||||||
|
|
||||||
/* allocate memory for f2fs-specific super block info */
|
/* allocate memory for f2fs-specific super block info */
|
||||||
|
@ -1260,7 +1249,8 @@ try_onemore:
|
||||||
goto free_sbi;
|
goto free_sbi;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = read_raw_super_block(sb, &raw_super, &raw_super_buf, &recovery);
|
err = read_raw_super_block(sb, &raw_super, &valid_super_block,
|
||||||
|
&recovery);
|
||||||
if (err)
|
if (err)
|
||||||
goto free_sbi;
|
goto free_sbi;
|
||||||
|
|
||||||
|
@ -1293,7 +1283,7 @@ try_onemore:
|
||||||
/* init f2fs-specific super block info */
|
/* init f2fs-specific super block info */
|
||||||
sbi->sb = sb;
|
sbi->sb = sb;
|
||||||
sbi->raw_super = raw_super;
|
sbi->raw_super = raw_super;
|
||||||
sbi->raw_super_buf = raw_super_buf;
|
sbi->valid_super_block = valid_super_block;
|
||||||
mutex_init(&sbi->gc_mutex);
|
mutex_init(&sbi->gc_mutex);
|
||||||
mutex_init(&sbi->writepages);
|
mutex_init(&sbi->writepages);
|
||||||
mutex_init(&sbi->cp_mutex);
|
mutex_init(&sbi->cp_mutex);
|
||||||
|
@ -1499,7 +1489,6 @@ free_meta_inode:
|
||||||
free_options:
|
free_options:
|
||||||
kfree(options);
|
kfree(options);
|
||||||
free_sb_buf:
|
free_sb_buf:
|
||||||
brelse(raw_super_buf);
|
|
||||||
kfree(raw_super);
|
kfree(raw_super);
|
||||||
free_sbi:
|
free_sbi:
|
||||||
kfree(sbi);
|
kfree(sbi);
|
||||||
|
|
Loading…
Reference in a new issue