Skip to content

Commit a12b877

Browse files
Qu Wenruokdave
authored andcommitted
btrfs: qgroup: Fix qgroup reserved space underflow caused by buffered write and quotas being enabled
[BUG] Under the following case, we can underflow qgroup reserved space. Task A | Task B --------------------------------------------------------------- Quota disabled | Buffered write | |- btrfs_check_data_free_space() | | *NO* qgroup space is reserved | | since quota is *DISABLED* | |- All pages are copied to page | cache | | Enable quota | Quota scan finished | | Sync_fs | |- run_delalloc_range | |- Write pages | |- btrfs_finish_ordered_io | |- insert_reserved_file_extent | |- btrfs_qgroup_release_data() | Since no qgroup space is reserved in Task A, we underflow qgroup reserved space This can be detected by fstest btrfs/104. [CAUSE] In insert_reserved_file_extent() we tell qgroup to release the @ram_bytes size of qgroup reserved_space in all cases. And btrfs_qgroup_release_data() will check if quotas are enabled. However in the above case, the buffered write happens before quota is enabled, so we don't have the reserved space for that range. [FIX] In insert_reserved_file_extent(), we tell qgroup to release the acctual byte number it released. In the above case, since we don't have the reserved space, we tell qgroups to release 0 byte, so the problem can be fixed. And thanks to the @reserved parameter introduced by the qgroup rework, and previous patch to return released bytes, the fix can be as small as 10 lines. Signed-off-by: Qu Wenruo <quwenruo@cn.fujitsu.com> [ changelog updates ] Signed-off-by: David Sterba <dsterba@suse.com>
1 parent 7bc329c commit a12b877

File tree

1 file changed

+8
-3
lines changed

1 file changed

+8
-3
lines changed

fs/btrfs/inode.c

Lines changed: 8 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -2145,6 +2145,7 @@ static int insert_reserved_file_extent(struct btrfs_trans_handle *trans,
21452145
struct btrfs_path *path;
21462146
struct extent_buffer *leaf;
21472147
struct btrfs_key ins;
2148+
u64 qg_released;
21482149
int extent_inserted = 0;
21492150
int ret;
21502151

@@ -2200,13 +2201,17 @@ static int insert_reserved_file_extent(struct btrfs_trans_handle *trans,
22002201
ins.objectid = disk_bytenr;
22012202
ins.offset = disk_num_bytes;
22022203
ins.type = BTRFS_EXTENT_ITEM_KEY;
2203-
ret = btrfs_alloc_reserved_file_extent(trans, root->root_key.objectid,
2204-
btrfs_ino(BTRFS_I(inode)), file_pos, ram_bytes, &ins);
2204+
22052205
/*
22062206
* Release the reserved range from inode dirty range map, as it is
22072207
* already moved into delayed_ref_head
22082208
*/
2209-
btrfs_qgroup_release_data(inode, file_pos, ram_bytes);
2209+
ret = btrfs_qgroup_release_data(inode, file_pos, ram_bytes);
2210+
if (ret < 0)
2211+
goto out;
2212+
qg_released = ret;
2213+
ret = btrfs_alloc_reserved_file_extent(trans, root->root_key.objectid,
2214+
btrfs_ino(BTRFS_I(inode)), file_pos, qg_released, &ins);
22102215
out:
22112216
btrfs_free_path(path);
22122217

0 commit comments

Comments
 (0)