Commit 6666e6aa authored by Jaegeuk Kim's avatar Jaegeuk Kim

f2fs: fix tracking parent inode number

Previously, f2fs didn't track the parent inode number correctly which is stored
in each f2fs_inode. In the case of the following scenario, a bug can be occured.

Let's suppose there are one directory, "/b", and two files, "/a" and "/b/a".
 - pino of "/a" is ROOT_INO.
 - pino of "/b/a" is DIR_B_INO.

Then,
 # sync
  : The inode pages of "/a" and "/b/a" contain the parent inode numbers as
    ROOT_INO and DIR_B_INO respectively.
 # mv /a /b/a
  : The parent inode number of "/a" should be changed to DIR_B_INO, but f2fs
    didn't do that. Ref. f2fs_set_link().

In order to fix this clearly, I added i_pino in f2fs_inode_info, and whenever
it needs to be changed like in f2fs_add_link() and f2fs_set_link(), it is
updated temporarily in f2fs_inode_info.

And later, f2fs_write_inode() stores the latest information to the inode pages.
For power-off-recovery, f2fs_sync_file() triggers simply f2fs_write_inode().
Signed-off-by: default avatarJaegeuk Kim <jaegeuk.kim@samsung.com>
parent 3cd8a239
...@@ -256,13 +256,16 @@ void f2fs_set_link(struct inode *dir, struct f2fs_dir_entry *de, ...@@ -256,13 +256,16 @@ void f2fs_set_link(struct inode *dir, struct f2fs_dir_entry *de,
set_page_dirty(page); set_page_dirty(page);
dir->i_mtime = dir->i_ctime = CURRENT_TIME; dir->i_mtime = dir->i_ctime = CURRENT_TIME;
mark_inode_dirty(dir); mark_inode_dirty(dir);
/* update parent inode number before releasing dentry page */
F2FS_I(inode)->i_pino = dir->i_ino;
f2fs_put_page(page, 1); f2fs_put_page(page, 1);
mutex_unlock_op(sbi, DENTRY_OPS); mutex_unlock_op(sbi, DENTRY_OPS);
} }
void init_dent_inode(struct dentry *dentry, struct page *ipage) void init_dent_inode(struct dentry *dentry, struct page *ipage)
{ {
struct inode *dir = dentry->d_parent->d_inode;
struct f2fs_node *rn; struct f2fs_node *rn;
if (IS_ERR(ipage)) if (IS_ERR(ipage))
...@@ -272,7 +275,6 @@ void init_dent_inode(struct dentry *dentry, struct page *ipage) ...@@ -272,7 +275,6 @@ void init_dent_inode(struct dentry *dentry, struct page *ipage)
/* copy dentry info. to this inode page */ /* copy dentry info. to this inode page */
rn = (struct f2fs_node *)page_address(ipage); rn = (struct f2fs_node *)page_address(ipage);
rn->i.i_pino = cpu_to_le32(dir->i_ino);
rn->i.i_namelen = cpu_to_le32(dentry->d_name.len); rn->i.i_namelen = cpu_to_le32(dentry->d_name.len);
memcpy(rn->i.i_name, dentry->d_name.name, dentry->d_name.len); memcpy(rn->i.i_name, dentry->d_name.name, dentry->d_name.len);
set_page_dirty(ipage); set_page_dirty(ipage);
...@@ -444,7 +446,11 @@ int f2fs_add_link(struct dentry *dentry, struct inode *inode) ...@@ -444,7 +446,11 @@ int f2fs_add_link(struct dentry *dentry, struct inode *inode)
for (i = 0; i < slots; i++) for (i = 0; i < slots; i++)
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); update_parent_metadata(dir, inode, current_depth);
/* update parent inode number before releasing dentry page */
F2FS_I(inode)->i_pino = dir->i_ino;
fail: fail:
kunmap(dentry_page); kunmap(dentry_page);
f2fs_put_page(dentry_page, 1); f2fs_put_page(dentry_page, 1);
......
...@@ -136,6 +136,7 @@ struct f2fs_inode_info { ...@@ -136,6 +136,7 @@ struct f2fs_inode_info {
unsigned long i_flags; /* keep an inode flags for ioctl */ unsigned long i_flags; /* keep an inode flags for ioctl */
unsigned char i_advise; /* use to give file attribute hints */ unsigned char i_advise; /* use to give file attribute hints */
unsigned int i_current_depth; /* use only in directory structure */ unsigned int i_current_depth; /* use only in directory structure */
unsigned int i_pino; /* parent inode number */
umode_t i_acl_mode; /* keep file acl mode temporarily */ umode_t i_acl_mode; /* keep file acl mode temporarily */
/* Use below internally in f2fs*/ /* Use below internally in f2fs*/
......
...@@ -107,6 +107,7 @@ static int do_read_inode(struct inode *inode) ...@@ -107,6 +107,7 @@ static int do_read_inode(struct inode *inode)
fi->flags = 0; fi->flags = 0;
fi->data_version = le64_to_cpu(F2FS_CKPT(sbi)->checkpoint_ver) - 1; fi->data_version = le64_to_cpu(F2FS_CKPT(sbi)->checkpoint_ver) - 1;
fi->i_advise = ri->i_advise; fi->i_advise = ri->i_advise;
fi->i_pino = le32_to_cpu(ri->i_pino);
get_extent_info(&fi->ext, ri->i_ext); get_extent_info(&fi->ext, ri->i_ext);
f2fs_put_page(node_page, 1); f2fs_put_page(node_page, 1);
return 0; return 0;
...@@ -200,6 +201,7 @@ void update_inode(struct inode *inode, struct page *node_page) ...@@ -200,6 +201,7 @@ void update_inode(struct inode *inode, struct page *node_page)
ri->i_current_depth = cpu_to_le32(F2FS_I(inode)->i_current_depth); ri->i_current_depth = cpu_to_le32(F2FS_I(inode)->i_current_depth);
ri->i_xattr_nid = cpu_to_le32(F2FS_I(inode)->i_xattr_nid); ri->i_xattr_nid = cpu_to_le32(F2FS_I(inode)->i_xattr_nid);
ri->i_flags = cpu_to_le32(F2FS_I(inode)->i_flags); ri->i_flags = cpu_to_le32(F2FS_I(inode)->i_flags);
ri->i_pino = cpu_to_le32(F2FS_I(inode)->i_pino);
ri->i_generation = cpu_to_le32(inode->i_generation); ri->i_generation = cpu_to_le32(inode->i_generation);
set_page_dirty(node_page); set_page_dirty(node_page);
} }
......
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