mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/kdave/btrfs-progs.git
synced 2025-01-26 13:44:06 +08:00
btrfs-progs: handle transaction start failure in close_ctree
Closing the fs will try to commit a pending transaction, but may fail to do so if the filesystem state is not well defined. This will eg. fail for some fuzz tests. The data structures are freed but no furhter attempt to commit is made. Signed-off-by: David Sterba <dsterba@suse.com>
This commit is contained in:
parent
c4dd5fef8a
commit
c6487a7d1d
12
disk-io.c
12
disk-io.c
@ -1623,6 +1623,7 @@ int write_ctree_super(struct btrfs_trans_handle *trans,
|
|||||||
int close_ctree_fs_info(struct btrfs_fs_info *fs_info)
|
int close_ctree_fs_info(struct btrfs_fs_info *fs_info)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
int err = 0;
|
||||||
struct btrfs_trans_handle *trans;
|
struct btrfs_trans_handle *trans;
|
||||||
struct btrfs_root *root = fs_info->tree_root;
|
struct btrfs_root *root = fs_info->tree_root;
|
||||||
|
|
||||||
@ -1630,7 +1631,10 @@ int close_ctree_fs_info(struct btrfs_fs_info *fs_info)
|
|||||||
fs_info->generation) {
|
fs_info->generation) {
|
||||||
BUG_ON(!root);
|
BUG_ON(!root);
|
||||||
trans = btrfs_start_transaction(root, 1);
|
trans = btrfs_start_transaction(root, 1);
|
||||||
BUG_ON(IS_ERR(trans));
|
if (IS_ERR(trans)) {
|
||||||
|
err = PTR_ERR(trans);
|
||||||
|
goto skip_commit;
|
||||||
|
}
|
||||||
btrfs_commit_transaction(trans, root);
|
btrfs_commit_transaction(trans, root);
|
||||||
trans = btrfs_start_transaction(root, 1);
|
trans = btrfs_start_transaction(root, 1);
|
||||||
BUG_ON(IS_ERR(trans));
|
BUG_ON(IS_ERR(trans));
|
||||||
@ -1650,6 +1654,8 @@ int close_ctree_fs_info(struct btrfs_fs_info *fs_info)
|
|||||||
fprintf(stderr,
|
fprintf(stderr,
|
||||||
"failed to write new super block err %d\n", ret);
|
"failed to write new super block err %d\n", ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
skip_commit:
|
||||||
btrfs_free_block_groups(fs_info);
|
btrfs_free_block_groups(fs_info);
|
||||||
|
|
||||||
free_fs_roots_tree(&fs_info->fs_root_tree);
|
free_fs_roots_tree(&fs_info->fs_root_tree);
|
||||||
@ -1658,7 +1664,9 @@ int close_ctree_fs_info(struct btrfs_fs_info *fs_info)
|
|||||||
ret = btrfs_close_devices(fs_info->fs_devices);
|
ret = btrfs_close_devices(fs_info->fs_devices);
|
||||||
btrfs_cleanup_all_caches(fs_info);
|
btrfs_cleanup_all_caches(fs_info);
|
||||||
btrfs_free_fs_info(fs_info);
|
btrfs_free_fs_info(fs_info);
|
||||||
return ret;
|
if (!err)
|
||||||
|
err = ret;
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
int clean_tree_block(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
int clean_tree_block(struct btrfs_trans_handle *trans, struct btrfs_root *root,
|
||||||
|
Loading…
Reference in New Issue
Block a user