mirror of https://gitee.com/openkylin/linux.git
f2fs: fix double lock for inode page during roll-foward recovery
If the inode is same and its data index are needed to truncate, we can fall into double lock for its inode page via get_dnode_of_data. Error case is like this. 1. write data 1, 2, 3, 4, 5 in inode #4. 2. write data 100, 102, 103, 104, 105 in dnode #6 of inode #4. 3. sync 4. update data 100->106 in dnode #6. 5. fsync inode #4. 6. power-cut -> Then, 1. go back to #3's checkpoint 2. in do_recover_data, get_dnode_of_data() gets inode #4. 3. detect 100->106 in dnode #6. 4. check_index_in_prev_nodes tries to truncate 100 in dnode #6. 5. to trigger truncate_hole, get_dnode_of_data should grab inode #4. 6. detect *kernel hang* This patch should resolve that bug. Signed-off-by: Jaegeuk Kim <jaegeuk@kernel.org>
This commit is contained in:
parent
c6e489305e
commit
60979115a6
|
@ -279,16 +279,30 @@ static int check_index_in_prev_nodes(struct f2fs_sb_info *sbi,
|
||||||
ino = ino_of_node(node_page);
|
ino = ino_of_node(node_page);
|
||||||
f2fs_put_page(node_page, 1);
|
f2fs_put_page(node_page, 1);
|
||||||
|
|
||||||
/* Deallocate previous index in the node page */
|
if (ino != dn->inode->i_ino) {
|
||||||
inode = f2fs_iget(sbi->sb, ino);
|
/* Deallocate previous index in the node page */
|
||||||
if (IS_ERR(inode))
|
inode = f2fs_iget(sbi->sb, ino);
|
||||||
return PTR_ERR(inode);
|
if (IS_ERR(inode))
|
||||||
|
return PTR_ERR(inode);
|
||||||
|
} else {
|
||||||
|
inode = dn->inode;
|
||||||
|
}
|
||||||
|
|
||||||
bidx = start_bidx_of_node(offset, F2FS_I(inode)) +
|
bidx = start_bidx_of_node(offset, F2FS_I(inode)) +
|
||||||
le16_to_cpu(sum.ofs_in_node);
|
le16_to_cpu(sum.ofs_in_node);
|
||||||
|
|
||||||
truncate_hole(inode, bidx, bidx + 1);
|
if (ino != dn->inode->i_ino) {
|
||||||
iput(inode);
|
truncate_hole(inode, bidx, bidx + 1);
|
||||||
|
iput(inode);
|
||||||
|
} else {
|
||||||
|
struct dnode_of_data tdn;
|
||||||
|
set_new_dnode(&tdn, inode, dn->inode_page, NULL, 0);
|
||||||
|
if (get_dnode_of_data(&tdn, bidx, LOOKUP_NODE))
|
||||||
|
return 0;
|
||||||
|
if (tdn.data_blkaddr != NULL_ADDR)
|
||||||
|
truncate_data_blocks_range(&tdn, 1);
|
||||||
|
f2fs_put_page(tdn.node_page, 1);
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue