Commit 50b21d7a authored by Christoph Hellwig's avatar Christoph Hellwig Committed by David Sterba

btrfs: submit a writeback bio per extent_buffer

Stop trying to cluster writes of multiple extent_buffers into a single
bio.  There is no need for that as the blk_plug mechanism used all the
way up in writeback_inodes_wb gives us the same I/O pattern even with
multiple bios.  Removing the clustering simplifies
lock_extent_buffer_for_io a lot and will also allow passing the eb
as private data to the end I/O handler.
Reviewed-by: default avatarJohannes Thumshirn <johannes.thumshirn@wdc.com>
Reviewed-by: default avatarJosef Bacik <josef@toxicpanda.com>
Signed-off-by: default avatarChristoph Hellwig <hch@lst.de>
Reviewed-by: default avatarDavid Sterba <dsterba@suse.com>
Signed-off-by: default avatarDavid Sterba <dsterba@suse.com>
parent 9fdd1601
...@@ -1627,41 +1627,24 @@ static void end_extent_buffer_writeback(struct extent_buffer *eb) ...@@ -1627,41 +1627,24 @@ static void end_extent_buffer_writeback(struct extent_buffer *eb)
/* /*
* Lock extent buffer status and pages for writeback. * Lock extent buffer status and pages for writeback.
* *
* May try to flush write bio if we can't get the lock.
*
* Return %false if the extent buffer doesn't need to be submitted (e.g. the * Return %false if the extent buffer doesn't need to be submitted (e.g. the
* extent buffer is not dirty) * extent buffer is not dirty)
* Return %true is the extent buffer is submitted to bio. * Return %true is the extent buffer is submitted to bio.
*/ */
static noinline_for_stack bool lock_extent_buffer_for_io(struct extent_buffer *eb, static noinline_for_stack bool lock_extent_buffer_for_io(struct extent_buffer *eb,
struct btrfs_bio_ctrl *bio_ctrl) struct writeback_control *wbc)
{ {
struct btrfs_fs_info *fs_info = eb->fs_info; struct btrfs_fs_info *fs_info = eb->fs_info;
int i, num_pages;
int flush = 0;
bool ret = false; bool ret = false;
int i;
if (!btrfs_try_tree_write_lock(eb)) { btrfs_tree_lock(eb);
submit_write_bio(bio_ctrl, 0); while (test_bit(EXTENT_BUFFER_WRITEBACK, &eb->bflags)) {
flush = 1;
btrfs_tree_lock(eb);
}
if (test_bit(EXTENT_BUFFER_WRITEBACK, &eb->bflags)) {
btrfs_tree_unlock(eb); btrfs_tree_unlock(eb);
if (bio_ctrl->wbc->sync_mode != WB_SYNC_ALL) if (wbc->sync_mode != WB_SYNC_ALL)
return false; return false;
if (!flush) { wait_on_extent_buffer_writeback(eb);
submit_write_bio(bio_ctrl, 0); btrfs_tree_lock(eb);
flush = 1;
}
while (1) {
wait_on_extent_buffer_writeback(eb);
btrfs_tree_lock(eb);
if (!test_bit(EXTENT_BUFFER_WRITEBACK, &eb->bflags))
break;
btrfs_tree_unlock(eb);
}
} }
/* /*
...@@ -1693,19 +1676,8 @@ static noinline_for_stack bool lock_extent_buffer_for_io(struct extent_buffer *e ...@@ -1693,19 +1676,8 @@ static noinline_for_stack bool lock_extent_buffer_for_io(struct extent_buffer *e
if (!ret || fs_info->nodesize < PAGE_SIZE) if (!ret || fs_info->nodesize < PAGE_SIZE)
return ret; return ret;
num_pages = num_extent_pages(eb); for (i = 0; i < num_extent_pages(eb); i++)
for (i = 0; i < num_pages; i++) { lock_page(eb->pages[i]);
struct page *p = eb->pages[i];
if (!trylock_page(p)) {
if (!flush) {
submit_write_bio(bio_ctrl, 0);
flush = 1;
}
lock_page(p);
}
}
return ret; return ret;
} }
...@@ -1936,11 +1908,16 @@ static void prepare_eb_write(struct extent_buffer *eb) ...@@ -1936,11 +1908,16 @@ static void prepare_eb_write(struct extent_buffer *eb)
* Page locking is only utilized at minimum to keep the VMM code happy. * Page locking is only utilized at minimum to keep the VMM code happy.
*/ */
static void write_one_subpage_eb(struct extent_buffer *eb, static void write_one_subpage_eb(struct extent_buffer *eb,
struct btrfs_bio_ctrl *bio_ctrl) struct writeback_control *wbc)
{ {
struct btrfs_fs_info *fs_info = eb->fs_info; struct btrfs_fs_info *fs_info = eb->fs_info;
struct page *page = eb->pages[0]; struct page *page = eb->pages[0];
bool no_dirty_ebs = false; bool no_dirty_ebs = false;
struct btrfs_bio_ctrl bio_ctrl = {
.wbc = wbc,
.opf = REQ_OP_WRITE | wbc_to_write_flags(wbc),
.end_io_func = end_bio_subpage_eb_writepage,
};
prepare_eb_write(eb); prepare_eb_write(eb);
...@@ -1954,40 +1931,43 @@ static void write_one_subpage_eb(struct extent_buffer *eb, ...@@ -1954,40 +1931,43 @@ static void write_one_subpage_eb(struct extent_buffer *eb,
if (no_dirty_ebs) if (no_dirty_ebs)
clear_page_dirty_for_io(page); clear_page_dirty_for_io(page);
bio_ctrl->end_io_func = end_bio_subpage_eb_writepage; submit_extent_page(&bio_ctrl, eb->start, page, eb->len,
submit_extent_page(bio_ctrl, eb->start, page, eb->len,
eb->start - page_offset(page)); eb->start - page_offset(page));
unlock_page(page); unlock_page(page);
submit_one_bio(&bio_ctrl);
/* /*
* Submission finished without problem, if no range of the page is * Submission finished without problem, if no range of the page is
* dirty anymore, we have submitted a page. Update nr_written in wbc. * dirty anymore, we have submitted a page. Update nr_written in wbc.
*/ */
if (no_dirty_ebs) if (no_dirty_ebs)
bio_ctrl->wbc->nr_to_write--; wbc->nr_to_write--;
} }
static noinline_for_stack void write_one_eb(struct extent_buffer *eb, static noinline_for_stack void write_one_eb(struct extent_buffer *eb,
struct btrfs_bio_ctrl *bio_ctrl) struct writeback_control *wbc)
{ {
u64 disk_bytenr = eb->start; u64 disk_bytenr = eb->start;
int i, num_pages; int i, num_pages;
struct btrfs_bio_ctrl bio_ctrl = {
.wbc = wbc,
.opf = REQ_OP_WRITE | wbc_to_write_flags(wbc),
.end_io_func = end_bio_extent_buffer_writepage,
};
prepare_eb_write(eb); prepare_eb_write(eb);
bio_ctrl->end_io_func = end_bio_extent_buffer_writepage;
num_pages = num_extent_pages(eb); num_pages = num_extent_pages(eb);
for (i = 0; i < num_pages; i++) { for (i = 0; i < num_pages; i++) {
struct page *p = eb->pages[i]; struct page *p = eb->pages[i];
clear_page_dirty_for_io(p); clear_page_dirty_for_io(p);
set_page_writeback(p); set_page_writeback(p);
submit_extent_page(bio_ctrl, disk_bytenr, p, PAGE_SIZE, 0); submit_extent_page(&bio_ctrl, disk_bytenr, p, PAGE_SIZE, 0);
disk_bytenr += PAGE_SIZE; disk_bytenr += PAGE_SIZE;
bio_ctrl->wbc->nr_to_write--; wbc->nr_to_write--;
unlock_page(p); unlock_page(p);
} }
submit_one_bio(&bio_ctrl);
} }
/* /*
...@@ -2004,7 +1984,7 @@ static noinline_for_stack void write_one_eb(struct extent_buffer *eb, ...@@ -2004,7 +1984,7 @@ static noinline_for_stack void write_one_eb(struct extent_buffer *eb,
* Return >=0 for the number of submitted extent buffers. * Return >=0 for the number of submitted extent buffers.
* Return <0 for fatal error. * Return <0 for fatal error.
*/ */
static int submit_eb_subpage(struct page *page, struct btrfs_bio_ctrl *bio_ctrl) static int submit_eb_subpage(struct page *page, struct writeback_control *wbc)
{ {
struct btrfs_fs_info *fs_info = btrfs_sb(page->mapping->host->i_sb); struct btrfs_fs_info *fs_info = btrfs_sb(page->mapping->host->i_sb);
int submitted = 0; int submitted = 0;
...@@ -2056,8 +2036,8 @@ static int submit_eb_subpage(struct page *page, struct btrfs_bio_ctrl *bio_ctrl) ...@@ -2056,8 +2036,8 @@ static int submit_eb_subpage(struct page *page, struct btrfs_bio_ctrl *bio_ctrl)
if (!eb) if (!eb)
continue; continue;
if (lock_extent_buffer_for_io(eb, bio_ctrl)) { if (lock_extent_buffer_for_io(eb, wbc)) {
write_one_subpage_eb(eb, bio_ctrl); write_one_subpage_eb(eb, wbc);
submitted++; submitted++;
} }
free_extent_buffer(eb); free_extent_buffer(eb);
...@@ -2085,7 +2065,7 @@ static int submit_eb_subpage(struct page *page, struct btrfs_bio_ctrl *bio_ctrl) ...@@ -2085,7 +2065,7 @@ static int submit_eb_subpage(struct page *page, struct btrfs_bio_ctrl *bio_ctrl)
* previous call. * previous call.
* Return <0 for fatal error. * Return <0 for fatal error.
*/ */
static int submit_eb_page(struct page *page, struct btrfs_bio_ctrl *bio_ctrl, static int submit_eb_page(struct page *page, struct writeback_control *wbc,
struct extent_buffer **eb_context) struct extent_buffer **eb_context)
{ {
struct address_space *mapping = page->mapping; struct address_space *mapping = page->mapping;
...@@ -2097,7 +2077,7 @@ static int submit_eb_page(struct page *page, struct btrfs_bio_ctrl *bio_ctrl, ...@@ -2097,7 +2077,7 @@ static int submit_eb_page(struct page *page, struct btrfs_bio_ctrl *bio_ctrl,
return 0; return 0;
if (btrfs_sb(page->mapping->host->i_sb)->nodesize < PAGE_SIZE) if (btrfs_sb(page->mapping->host->i_sb)->nodesize < PAGE_SIZE)
return submit_eb_subpage(page, bio_ctrl); return submit_eb_subpage(page, wbc);
spin_lock(&mapping->private_lock); spin_lock(&mapping->private_lock);
if (!PagePrivate(page)) { if (!PagePrivate(page)) {
...@@ -2130,8 +2110,7 @@ static int submit_eb_page(struct page *page, struct btrfs_bio_ctrl *bio_ctrl, ...@@ -2130,8 +2110,7 @@ static int submit_eb_page(struct page *page, struct btrfs_bio_ctrl *bio_ctrl,
* If for_sync, this hole will be filled with * If for_sync, this hole will be filled with
* trasnsaction commit. * trasnsaction commit.
*/ */
if (bio_ctrl->wbc->sync_mode == WB_SYNC_ALL && if (wbc->sync_mode == WB_SYNC_ALL && !wbc->for_sync)
!bio_ctrl->wbc->for_sync)
ret = -EAGAIN; ret = -EAGAIN;
else else
ret = 0; ret = 0;
...@@ -2141,12 +2120,12 @@ static int submit_eb_page(struct page *page, struct btrfs_bio_ctrl *bio_ctrl, ...@@ -2141,12 +2120,12 @@ static int submit_eb_page(struct page *page, struct btrfs_bio_ctrl *bio_ctrl,
*eb_context = eb; *eb_context = eb;
if (!lock_extent_buffer_for_io(eb, bio_ctrl)) { if (!lock_extent_buffer_for_io(eb, wbc)) {
btrfs_revert_meta_write_pointer(cache, eb); btrfs_revert_meta_write_pointer(cache, eb);
if (cache) if (cache)
btrfs_put_block_group(cache); btrfs_put_block_group(cache);
free_extent_buffer(eb); free_extent_buffer(eb);
return ret; return 0;
} }
if (cache) { if (cache) {
/* /*
...@@ -2155,7 +2134,7 @@ static int submit_eb_page(struct page *page, struct btrfs_bio_ctrl *bio_ctrl, ...@@ -2155,7 +2134,7 @@ static int submit_eb_page(struct page *page, struct btrfs_bio_ctrl *bio_ctrl,
btrfs_schedule_zone_finish_bg(cache, eb); btrfs_schedule_zone_finish_bg(cache, eb);
btrfs_put_block_group(cache); btrfs_put_block_group(cache);
} }
write_one_eb(eb, bio_ctrl); write_one_eb(eb, wbc);
free_extent_buffer(eb); free_extent_buffer(eb);
return 1; return 1;
} }
...@@ -2164,11 +2143,6 @@ int btree_write_cache_pages(struct address_space *mapping, ...@@ -2164,11 +2143,6 @@ int btree_write_cache_pages(struct address_space *mapping,
struct writeback_control *wbc) struct writeback_control *wbc)
{ {
struct extent_buffer *eb_context = NULL; struct extent_buffer *eb_context = NULL;
struct btrfs_bio_ctrl bio_ctrl = {
.wbc = wbc,
.opf = REQ_OP_WRITE | wbc_to_write_flags(wbc),
.extent_locked = 0,
};
struct btrfs_fs_info *fs_info = BTRFS_I(mapping->host)->root->fs_info; struct btrfs_fs_info *fs_info = BTRFS_I(mapping->host)->root->fs_info;
int ret = 0; int ret = 0;
int done = 0; int done = 0;
...@@ -2210,7 +2184,7 @@ int btree_write_cache_pages(struct address_space *mapping, ...@@ -2210,7 +2184,7 @@ int btree_write_cache_pages(struct address_space *mapping,
for (i = 0; i < nr_folios; i++) { for (i = 0; i < nr_folios; i++) {
struct folio *folio = fbatch.folios[i]; struct folio *folio = fbatch.folios[i];
ret = submit_eb_page(&folio->page, &bio_ctrl, &eb_context); ret = submit_eb_page(&folio->page, wbc, &eb_context);
if (ret == 0) if (ret == 0)
continue; continue;
if (ret < 0) { if (ret < 0) {
...@@ -2271,8 +2245,6 @@ int btree_write_cache_pages(struct address_space *mapping, ...@@ -2271,8 +2245,6 @@ int btree_write_cache_pages(struct address_space *mapping,
ret = 0; ret = 0;
if (!ret && BTRFS_FS_ERROR(fs_info)) if (!ret && BTRFS_FS_ERROR(fs_info))
ret = -EROFS; ret = -EROFS;
submit_write_bio(&bio_ctrl, ret);
btrfs_zoned_meta_io_unlock(fs_info); btrfs_zoned_meta_io_unlock(fs_info);
return ret; return ret;
} }
......
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