提交 ee0876bc 编写于 作者: J Jan Kara 提交者: Theodore Ts'o

ext4: do not ask jbd2 to write data for delalloc buffers

Currently we ask jbd2 to write all dirty allocated buffers before
committing a transaction when doing writeback of delay allocated blocks.
However this is unnecessary since we move all pages to writeback state
before dropping a transaction handle and then submit all the necessary
IO. We still need the transaction commit to wait for all the outstanding
writeback before flushing disk caches during transaction commit to avoid
data exposure issues though. Use the new jbd2 capability and ask it to
only wait for outstanding writeback during transaction commit when
writing back data in ext4_writepages().
Tested-by: N"HUANG Weller (CM/ESW12-CN)" <Weller.Huang@cn.bosch.com>
Signed-off-by: NJan Kara <jack@suse.cz>
Signed-off-by: NTheodore Ts'o <tytso@mit.edu>
上级 41617e1a
...@@ -581,6 +581,9 @@ enum { ...@@ -581,6 +581,9 @@ enum {
#define EXT4_GET_BLOCKS_ZERO 0x0200 #define EXT4_GET_BLOCKS_ZERO 0x0200
#define EXT4_GET_BLOCKS_CREATE_ZERO (EXT4_GET_BLOCKS_CREATE |\ #define EXT4_GET_BLOCKS_CREATE_ZERO (EXT4_GET_BLOCKS_CREATE |\
EXT4_GET_BLOCKS_ZERO) EXT4_GET_BLOCKS_ZERO)
/* Caller will submit data before dropping transaction handle. This
* allows jbd2 to avoid submitting data before commit. */
#define EXT4_GET_BLOCKS_IO_SUBMIT 0x0400
/* /*
* The bit position of these flags must not overlap with any of the * The bit position of these flags must not overlap with any of the
......
...@@ -359,7 +359,8 @@ static inline int ext4_journal_force_commit(journal_t *journal) ...@@ -359,7 +359,8 @@ static inline int ext4_journal_force_commit(journal_t *journal)
return 0; return 0;
} }
static inline int ext4_jbd2_file_inode(handle_t *handle, struct inode *inode) static inline int ext4_jbd2_inode_add_write(handle_t *handle,
struct inode *inode)
{ {
if (ext4_handle_valid(handle)) if (ext4_handle_valid(handle))
return jbd2_journal_inode_add_write(handle, return jbd2_journal_inode_add_write(handle,
...@@ -367,6 +368,15 @@ static inline int ext4_jbd2_file_inode(handle_t *handle, struct inode *inode) ...@@ -367,6 +368,15 @@ static inline int ext4_jbd2_file_inode(handle_t *handle, struct inode *inode)
return 0; return 0;
} }
static inline int ext4_jbd2_inode_add_wait(handle_t *handle,
struct inode *inode)
{
if (ext4_handle_valid(handle))
return jbd2_journal_inode_add_wait(handle,
EXT4_I(inode)->jinode);
return 0;
}
static inline void ext4_update_inode_fsync_trans(handle_t *handle, static inline void ext4_update_inode_fsync_trans(handle_t *handle,
struct inode *inode, struct inode *inode,
int datasync) int datasync)
......
...@@ -695,7 +695,10 @@ int ext4_map_blocks(handle_t *handle, struct inode *inode, ...@@ -695,7 +695,10 @@ int ext4_map_blocks(handle_t *handle, struct inode *inode,
!(flags & EXT4_GET_BLOCKS_ZERO) && !(flags & EXT4_GET_BLOCKS_ZERO) &&
!IS_NOQUOTA(inode) && !IS_NOQUOTA(inode) &&
ext4_should_order_data(inode)) { ext4_should_order_data(inode)) {
ret = ext4_jbd2_file_inode(handle, inode); if (flags & EXT4_GET_BLOCKS_IO_SUBMIT)
ret = ext4_jbd2_inode_add_wait(handle, inode);
else
ret = ext4_jbd2_inode_add_write(handle, inode);
if (ret) if (ret)
return ret; return ret;
} }
...@@ -2319,7 +2322,8 @@ static int mpage_map_one_extent(handle_t *handle, struct mpage_da_data *mpd) ...@@ -2319,7 +2322,8 @@ static int mpage_map_one_extent(handle_t *handle, struct mpage_da_data *mpd)
* the data was copied into the page cache. * the data was copied into the page cache.
*/ */
get_blocks_flags = EXT4_GET_BLOCKS_CREATE | get_blocks_flags = EXT4_GET_BLOCKS_CREATE |
EXT4_GET_BLOCKS_METADATA_NOFAIL; EXT4_GET_BLOCKS_METADATA_NOFAIL |
EXT4_GET_BLOCKS_IO_SUBMIT;
dioread_nolock = ext4_should_dioread_nolock(inode); dioread_nolock = ext4_should_dioread_nolock(inode);
if (dioread_nolock) if (dioread_nolock)
get_blocks_flags |= EXT4_GET_BLOCKS_IO_CREATE_EXT; get_blocks_flags |= EXT4_GET_BLOCKS_IO_CREATE_EXT;
...@@ -3634,7 +3638,7 @@ static int __ext4_block_zero_page_range(handle_t *handle, ...@@ -3634,7 +3638,7 @@ static int __ext4_block_zero_page_range(handle_t *handle,
err = 0; err = 0;
mark_buffer_dirty(bh); mark_buffer_dirty(bh);
if (ext4_should_order_data(inode)) if (ext4_should_order_data(inode))
err = ext4_jbd2_file_inode(handle, inode); err = ext4_jbd2_inode_add_write(handle, inode);
} }
unlock: unlock:
......
...@@ -400,7 +400,7 @@ move_extent_per_page(struct file *o_filp, struct inode *donor_inode, ...@@ -400,7 +400,7 @@ move_extent_per_page(struct file *o_filp, struct inode *donor_inode,
/* Even in case of data=writeback it is reasonable to pin /* Even in case of data=writeback it is reasonable to pin
* inode to transaction, to prevent unexpected data loss */ * inode to transaction, to prevent unexpected data loss */
*err = ext4_jbd2_file_inode(handle, orig_inode); *err = ext4_jbd2_inode_add_write(handle, orig_inode);
unlock_pages: unlock_pages:
unlock_page(pagep[0]); unlock_page(pagep[0]);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册