From: Zheng Liu Date: Mon, 3 Mar 2014 05:27:48 +0000 (-0500) Subject: libext2fs: handle inline_data in block iterator function X-Git-Tag: v1.43-WIP-2015-05-18~345 X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=716a3c3536f87372ce16b5b678b9957dabf6d638;p=thirdparty%2Fe2fsprogs.git libext2fs: handle inline_data in block iterator function After applied this commit (a7f4c635), we have banned to traverse blocks for an inode which has inline data because no block belongs to it. But before calling this function, we need to check inline data flag. This commit add a sanity check ext2fs_inode_has_valid_blocks2() to fix them except that ext2fs_expand_dir because it will be fixed by another patch. Meanwhile in this commit it fixes a bug that when we kill a file we could leak an inode. Signed-off-by: Zheng Liu Signed-off-by: "Theodore Ts'o" --- diff --git a/debugfs/debugfs.c b/debugfs/debugfs.c index a3fcbf417..29c016162 100644 --- a/debugfs/debugfs.c +++ b/debugfs/debugfs.c @@ -1924,11 +1924,10 @@ static void kill_file_by_inode(ext2_ino_t inode) inode_buf.i_dtime = current_fs->now ? current_fs->now : time(0); if (debugfs_write_inode(inode, &inode_buf, 0)) return; - if (!ext2fs_inode_has_valid_blocks2(current_fs, &inode_buf)) - return; - - ext2fs_block_iterate3(current_fs, inode, BLOCK_FLAG_READ_ONLY, NULL, - release_blocks_proc, NULL); + if (ext2fs_inode_has_valid_blocks2(current_fs, &inode_buf)) { + ext2fs_block_iterate3(current_fs, inode, BLOCK_FLAG_READ_ONLY, + NULL, release_blocks_proc, NULL); + } printf("\n"); ext2fs_inode_alloc_stats2(current_fs, inode, -1, LINUX_S_ISDIR(inode_buf.i_mode)); diff --git a/debugfs/filefrag.c b/debugfs/filefrag.c index 6219d7cb7..49345ad1f 100644 --- a/debugfs/filefrag.c +++ b/debugfs/filefrag.c @@ -154,11 +154,13 @@ static void filefrag(ext2_ino_t ino, struct ext2_inode *inode, fs->name, num_blocks, EXT2_I_SIZE(inode)); } print_header(fs); - retval = ext2fs_block_iterate3(current_fs, ino, - BLOCK_FLAG_READ_ONLY, NULL, - filefrag_blocks_proc, fs); - if (retval) - com_err("ext2fs_block_iterate3", retval, 0); + if (ext2fs_inode_has_valid_blocks2(current_fs, inode)) { + retval = ext2fs_block_iterate3(current_fs, ino, + BLOCK_FLAG_READ_ONLY, NULL, + filefrag_blocks_proc, fs); + if (retval) + com_err("ext2fs_block_iterate3", retval, 0); + } report_filefrag(fs); fprintf(fs->f, "%s: %d contiguous extents%s\n", fs->name, fs->ext, diff --git a/debugfs/lsdel.c b/debugfs/lsdel.c index e5b2d203b..0e774a302 100644 --- a/debugfs/lsdel.c +++ b/debugfs/lsdel.c @@ -141,13 +141,16 @@ void do_lsdel(int argc, char **argv) lsd.free_blocks = 0; lsd.bad_blocks = 0; - retval = ext2fs_block_iterate3(current_fs, ino, - BLOCK_FLAG_READ_ONLY, block_buf, - lsdel_proc, &lsd); - if (retval) { - com_err("ls_deleted_inodes", retval, - "while calling ext2fs_block_iterate2"); - goto next; + if (ext2fs_inode_has_valid_blocks2(current_fs, &inode)) { + retval = ext2fs_block_iterate3(current_fs, ino, + BLOCK_FLAG_READ_ONLY, + block_buf, + lsdel_proc, &lsd); + if (retval) { + com_err("ls_deleted_inodes", retval, + "while calling ext2fs_block_iterate2"); + goto next; + } } if (lsd.free_blocks && !lsd.bad_blocks) { if (num_delarray >= max_delarray) { diff --git a/lib/ext2fs/valid_blk.c b/lib/ext2fs/valid_blk.c index 895e36ef0..db5d90ae4 100644 --- a/lib/ext2fs/valid_blk.c +++ b/lib/ext2fs/valid_blk.c @@ -52,6 +52,13 @@ int ext2fs_inode_has_valid_blocks2(ext2_filsys fs, struct ext2_inode *inode) return 0; /* Probably a fast symlink */ } } + + /* + * If this inode has inline data, it shouldn't have valid block + * entries. + */ + if (inode->i_flags & EXT4_INLINE_DATA_FL) + return 0; return 1; } diff --git a/misc/tune2fs.c b/misc/tune2fs.c index 09dd973ae..657176415 100644 --- a/misc/tune2fs.c +++ b/misc/tune2fs.c @@ -693,7 +693,8 @@ static void rewrite_inodes(ext2_filsys fs) exit(1); } - if (LINUX_S_ISDIR(inode->i_mode)) { + if (LINUX_S_ISDIR(inode->i_mode) && + ext2fs_inode_has_valid_blocks2(fs, inode)) { retval = rewrite_directory(fs, ino, inode); if (retval) { com_err("rewrite_directory", retval,