Commit 44a83ff6 authored by Jaegeuk Kim's avatar Jaegeuk Kim

f2fs: update inode page after creation

I found a bug when testing power-off-recovery as follows.

[Bug Scenario]
1. create a file
2. fsync the file
3. reboot w/o any sync
4. try to recover the file
 - found its fsync mark
 - found its dentry mark
   : try to recover its dentry
    - get its file name
    - get its parent inode number
     : here we got zero value

The reason why we get the wrong parent inode number is that we didn't
synchronize the inode page with its newly created inode information perfectly.

Especially, previous f2fs stores fi->i_pino and writes it to the cached
node page in a wrong order, which incurs the zero-valued i_pino during the
recovery.

So, this patch modifies the creation flow to fix the synchronization order of
inode page with its inode.
Signed-off-by: default avatarJaegeuk Kim <jaegeuk.kim@samsung.com>
parent 64aa7ed9
...@@ -279,6 +279,7 @@ struct page *get_lock_data_page(struct inode *inode, pgoff_t index) ...@@ -279,6 +279,7 @@ struct page *get_lock_data_page(struct inode *inode, pgoff_t index)
* *
* Also, caller should grab and release a mutex by calling mutex_lock_op() and * Also, caller should grab and release a mutex by calling mutex_lock_op() and
* mutex_unlock_op(). * mutex_unlock_op().
* Note that, npage is set only by make_empty_dir.
*/ */
struct page *get_new_data_page(struct inode *inode, struct page *get_new_data_page(struct inode *inode,
struct page *npage, pgoff_t index, bool new_i_size) struct page *npage, pgoff_t index, bool new_i_size)
......
...@@ -264,15 +264,10 @@ void f2fs_set_link(struct inode *dir, struct f2fs_dir_entry *de, ...@@ -264,15 +264,10 @@ void f2fs_set_link(struct inode *dir, struct f2fs_dir_entry *de,
f2fs_put_page(page, 1); f2fs_put_page(page, 1);
} }
void init_dent_inode(const struct qstr *name, struct page *ipage) static void init_dent_inode(const struct qstr *name, struct page *ipage)
{ {
struct f2fs_node *rn; struct f2fs_node *rn;
if (IS_ERR(ipage))
return;
wait_on_page_writeback(ipage);
/* copy name info. to this inode page */ /* copy name info. to this inode page */
rn = (struct f2fs_node *)page_address(ipage); rn = (struct f2fs_node *)page_address(ipage);
rn->i.i_namelen = cpu_to_le32(name->len); rn->i.i_namelen = cpu_to_le32(name->len);
...@@ -280,14 +275,15 @@ void init_dent_inode(const struct qstr *name, struct page *ipage) ...@@ -280,14 +275,15 @@ void init_dent_inode(const struct qstr *name, struct page *ipage)
set_page_dirty(ipage); set_page_dirty(ipage);
} }
static int make_empty_dir(struct inode *inode, struct inode *parent) static int make_empty_dir(struct inode *inode,
struct inode *parent, struct page *page)
{ {
struct page *dentry_page; struct page *dentry_page;
struct f2fs_dentry_block *dentry_blk; struct f2fs_dentry_block *dentry_blk;
struct f2fs_dir_entry *de; struct f2fs_dir_entry *de;
void *kaddr; void *kaddr;
dentry_page = get_new_data_page(inode, NULL, 0, true); dentry_page = get_new_data_page(inode, page, 0, true);
if (IS_ERR(dentry_page)) if (IS_ERR(dentry_page))
return PTR_ERR(dentry_page); return PTR_ERR(dentry_page);
...@@ -317,42 +313,47 @@ static int make_empty_dir(struct inode *inode, struct inode *parent) ...@@ -317,42 +313,47 @@ static int make_empty_dir(struct inode *inode, struct inode *parent)
return 0; return 0;
} }
static int init_inode_metadata(struct inode *inode, static struct page *init_inode_metadata(struct inode *inode,
struct inode *dir, const struct qstr *name) struct inode *dir, const struct qstr *name)
{ {
struct page *page;
int err;
if (is_inode_flag_set(F2FS_I(inode), FI_NEW_INODE)) { if (is_inode_flag_set(F2FS_I(inode), FI_NEW_INODE)) {
int err; page = new_inode_page(inode, name);
err = new_inode_page(inode, name); if (IS_ERR(page))
if (err) return page;
return err;
if (S_ISDIR(inode->i_mode)) { if (S_ISDIR(inode->i_mode)) {
err = make_empty_dir(inode, dir); err = make_empty_dir(inode, dir, page);
if (err) { if (err)
remove_inode_page(inode); goto error;
return err;
}
} }
err = f2fs_init_acl(inode, dir); err = f2fs_init_acl(inode, dir);
if (err) { if (err)
remove_inode_page(inode); goto error;
return err;
} wait_on_page_writeback(page);
} else { } else {
struct page *ipage; page = get_node_page(F2FS_SB(dir->i_sb), inode->i_ino);
ipage = get_node_page(F2FS_SB(dir->i_sb), inode->i_ino); if (IS_ERR(page))
if (IS_ERR(ipage)) return page;
return PTR_ERR(ipage);
set_cold_node(inode, ipage); wait_on_page_writeback(page);
init_dent_inode(name, ipage); set_cold_node(inode, page);
f2fs_put_page(ipage, 1);
} }
if (is_inode_flag_set(F2FS_I(inode), FI_INC_LINK)) {
init_dent_inode(name, page);
if (is_inode_flag_set(F2FS_I(inode), FI_INC_LINK))
inc_nlink(inode); inc_nlink(inode);
update_inode_page(inode); return page;
}
return 0; error:
f2fs_put_page(page, 1);
remove_inode_page(inode);
return ERR_PTR(err);
} }
static void update_parent_metadata(struct inode *dir, struct inode *inode, static void update_parent_metadata(struct inode *dir, struct inode *inode,
...@@ -423,6 +424,7 @@ int __f2fs_add_link(struct inode *dir, const struct qstr *name, struct inode *in ...@@ -423,6 +424,7 @@ int __f2fs_add_link(struct inode *dir, const struct qstr *name, struct inode *in
struct page *dentry_page = NULL; struct page *dentry_page = NULL;
struct f2fs_dentry_block *dentry_blk = NULL; struct f2fs_dentry_block *dentry_blk = NULL;
int slots = GET_DENTRY_SLOTS(namelen); int slots = GET_DENTRY_SLOTS(namelen);
struct page *page;
int err = 0; int err = 0;
int i; int i;
...@@ -465,12 +467,13 @@ int __f2fs_add_link(struct inode *dir, const struct qstr *name, struct inode *in ...@@ -465,12 +467,13 @@ int __f2fs_add_link(struct inode *dir, const struct qstr *name, struct inode *in
++level; ++level;
goto start; goto start;
add_dentry: add_dentry:
err = init_inode_metadata(inode, dir, name);
if (err)
goto fail;
wait_on_page_writeback(dentry_page); wait_on_page_writeback(dentry_page);
page = init_inode_metadata(inode, dir, name);
if (IS_ERR(page)) {
err = PTR_ERR(page);
goto fail;
}
de = &dentry_blk->dentry[bit_pos]; de = &dentry_blk->dentry[bit_pos];
de->hash_code = dentry_hash; de->hash_code = dentry_hash;
de->name_len = cpu_to_le16(namelen); de->name_len = cpu_to_le16(namelen);
...@@ -481,10 +484,12 @@ int __f2fs_add_link(struct inode *dir, const struct qstr *name, struct inode *in ...@@ -481,10 +484,12 @@ int __f2fs_add_link(struct inode *dir, const struct qstr *name, struct inode *in
test_and_set_bit_le(bit_pos + i, &dentry_blk->dentry_bitmap); test_and_set_bit_le(bit_pos + i, &dentry_blk->dentry_bitmap);
set_page_dirty(dentry_page); set_page_dirty(dentry_page);
update_parent_metadata(dir, inode, current_depth); /* we don't need to mark_inode_dirty now */
/* update parent inode number before releasing dentry page */
F2FS_I(inode)->i_pino = dir->i_ino; F2FS_I(inode)->i_pino = dir->i_ino;
update_inode(inode, page);
f2fs_put_page(page, 1);
update_parent_metadata(dir, inode, current_depth);
fail: fail:
kunmap(dentry_page); kunmap(dentry_page);
f2fs_put_page(dentry_page, 1); f2fs_put_page(dentry_page, 1);
......
...@@ -914,7 +914,6 @@ struct f2fs_dir_entry *f2fs_parent_dir(struct inode *, struct page **); ...@@ -914,7 +914,6 @@ struct f2fs_dir_entry *f2fs_parent_dir(struct inode *, struct page **);
ino_t f2fs_inode_by_name(struct inode *, struct qstr *); ino_t f2fs_inode_by_name(struct inode *, struct qstr *);
void f2fs_set_link(struct inode *, struct f2fs_dir_entry *, void f2fs_set_link(struct inode *, struct f2fs_dir_entry *,
struct page *, struct inode *); struct page *, struct inode *);
void init_dent_inode(const struct qstr *, struct page *);
int __f2fs_add_link(struct inode *, const struct qstr *, struct inode *); int __f2fs_add_link(struct inode *, const struct qstr *, struct inode *);
void f2fs_delete_entry(struct f2fs_dir_entry *, struct page *, struct inode *); void f2fs_delete_entry(struct f2fs_dir_entry *, struct page *, struct inode *);
int f2fs_make_empty(struct inode *, struct inode *); int f2fs_make_empty(struct inode *, struct inode *);
...@@ -949,7 +948,7 @@ void get_node_info(struct f2fs_sb_info *, nid_t, struct node_info *); ...@@ -949,7 +948,7 @@ void get_node_info(struct f2fs_sb_info *, nid_t, struct node_info *);
int get_dnode_of_data(struct dnode_of_data *, pgoff_t, int); int get_dnode_of_data(struct dnode_of_data *, pgoff_t, int);
int truncate_inode_blocks(struct inode *, pgoff_t); int truncate_inode_blocks(struct inode *, pgoff_t);
int remove_inode_page(struct inode *); int remove_inode_page(struct inode *);
int new_inode_page(struct inode *, const struct qstr *); struct page *new_inode_page(struct inode *, const struct qstr *);
struct page *new_node_page(struct dnode_of_data *, unsigned int); struct page *new_node_page(struct dnode_of_data *, unsigned int);
void ra_node_page(struct f2fs_sb_info *, nid_t); void ra_node_page(struct f2fs_sb_info *, nid_t);
struct page *get_node_page(struct f2fs_sb_info *, pgoff_t); struct page *get_node_page(struct f2fs_sb_info *, pgoff_t);
......
...@@ -806,19 +806,15 @@ int remove_inode_page(struct inode *inode) ...@@ -806,19 +806,15 @@ int remove_inode_page(struct inode *inode)
return 0; return 0;
} }
int new_inode_page(struct inode *inode, const struct qstr *name) struct page *new_inode_page(struct inode *inode, const struct qstr *name)
{ {
struct page *page;
struct dnode_of_data dn; struct dnode_of_data dn;
/* allocate inode page for new inode */ /* allocate inode page for new inode */
set_new_dnode(&dn, inode, NULL, NULL, inode->i_ino); set_new_dnode(&dn, inode, NULL, NULL, inode->i_ino);
page = new_node_page(&dn, 0);
init_dent_inode(name, page); /* caller should f2fs_put_page(page, 1); */
if (IS_ERR(page)) return new_node_page(&dn, 0);
return PTR_ERR(page);
f2fs_put_page(page, 1);
return 0;
} }
struct page *new_node_page(struct dnode_of_data *dn, unsigned int ofs) struct page *new_node_page(struct dnode_of_data *dn, unsigned int ofs)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment